mirror of
https://github.com/NetSPI/PowerHuntShares.git
synced 2025-05-04 19:28:42 +02:00
15737 lines
791 KiB
PowerShell
15737 lines
791 KiB
PowerShell
#Requires -Version 5.1
|
|
#--------------------------------------
|
|
# Function: Invoke-HuntSMBShares
|
|
#--------------------------------------
|
|
# Author: Scott Sutherland, 2024 NetSPI
|
|
# License: 3-clause BSD
|
|
# Version: v1.49
|
|
# References: This script includes custom code and code taken and modified from the open source projects PowerView, Invoke-Ping, and Invoke-Parrell.
|
|
function Invoke-HuntSMBShares
|
|
{
|
|
<#
|
|
.SYNOPSIS
|
|
This function can be used to inventory to SMB shares on the current Active Directory domain and identify potentially high risk exposures.
|
|
It will automatically generate csv files and html summary report.
|
|
.PARAMETER Threads
|
|
Number of concurrent tasks to run at once.
|
|
.PARAMETER Output Directory
|
|
File path where all csv and html report will be exported.
|
|
.EXAMPLE
|
|
PS C:\temp\test> Invoke-HuntSMBShares -Threads 20 -OutputDirectory c:\temp\test -DomainController 10.1.1.1 -ExportFindings -Username domain\user -Password password
|
|
.EXAMPLE
|
|
C:\temp\test> runas /netonly /user:domain\user PowerShell.exe
|
|
PS C:\temp\test> Import-Module Invoke-HuntSMBShares.ps1
|
|
PS C:\temp\test> Invoke-HuntSMBShares -Threads 20 -RunSpaceTimeOut 10 -OutputDirectory c:\folder\ -DomainController 10.1.1.1 -ExportFindings -Username domain\user -Password password
|
|
.EXAMPLE
|
|
PS C:\temp\test> Invoke-HuntSMBShares -Threads 20 -ExportFindings -OutputDirectory c:\temp\test
|
|
|
|
---------------------------------------------------------------
|
|
INVOKE-HUNTSMBSHARES
|
|
---------------------------------------------------------------
|
|
This function automates the following tasks:
|
|
|
|
o Determine current computer's domain
|
|
o Enumerate domain computers
|
|
o Check if computers respond to ping requests
|
|
o Filter for computers that have TCP 445 open and accessible
|
|
o Enumerate SMB shares
|
|
o Enumerate SMB share permissions
|
|
o Identify shares with potentially excessive privileges
|
|
o Identify shares that provide read or write access
|
|
o Identify shares thare are high risk
|
|
o Identify common share owners, names, & directory listings
|
|
o Generate last written & last accessed timelines
|
|
o Generate html summary report and detailed csv files
|
|
|
|
Note: This can take hours to run in large environments.
|
|
---------------------------------------------------------------
|
|
|||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
|
---------------------------------------------------------------
|
|
SHARE DISCOVERY
|
|
---------------------------------------------------------------
|
|
[*][03/01/2021 09:35] Scan Start
|
|
[*][03/01/2021 09:35] Output Directory: c:\temp\smbshares\SmbShareHunt-03012021093504
|
|
[*][03/01/2021 09:35] Successful connection to domain controller: dc1.demo.local
|
|
[*][03/01/2021 09:35] Performing LDAP query for computers associated with the demo.local domain
|
|
[*][03/01/2021 09:35] - 245 computers found
|
|
[*][03/01/2021 09:35] Pinging 245 computers
|
|
[*][03/01/2021 09:35] - 55 computers responded to ping requests.
|
|
[*][03/01/2021 09:35] Checking if TCP Port 445 is open on 55 computers
|
|
[*][03/01/2021 09:36] - 49 computers have TCP port 445 open.
|
|
[*][03/01/2021 09:36] Getting a list of SMB shares from 49 computers
|
|
[*][03/01/2021 09:36] - 217 SMB shares were found.
|
|
[*][03/01/2021 09:36] Getting share permissions from 217 SMB shares
|
|
[*][03/01/2021 09:37] - 374 share permissions were enumerated.
|
|
[*][03/01/2021 09:37] Getting directory listings from 33 SMB shares
|
|
[*][03/01/2021 09:37] - Targeting up to 3 nested directory levels
|
|
[*][03/01/2021 09:37] - 563 files and folders were enumerated.
|
|
[*][03/01/2021 09:37] Identifying potentially excessive share permissions
|
|
[*][03/01/2021 09:37] - 33 potentially excessive privileges were found across 12 systems..
|
|
[*][03/01/2021 09:37] Scan Complete
|
|
---------------------------------------------------------------
|
|
SHARE ANALYSIS
|
|
---------------------------------------------------------------
|
|
[*][03/01/2021 09:37] Analysis Start
|
|
[*][03/01/2021 09:37] - 14 shares can be read across 12 systems.
|
|
[*][03/01/2021 09:37] - 1 shares can be written to across 1 systems.
|
|
[*][03/01/2021 09:37] - 46 shares are considered non-default across 32 systems.
|
|
[*][03/01/2021 09:37] - 0 shares are considered high risk across 0 systems
|
|
[*][03/01/2021 09:37] - Identified top 5 owners of excessive shares.
|
|
[*][03/01/2021 09:37] - Identified top 5 share groups.
|
|
[*][03/01/2021 09:37] - Identified top 5 share names.
|
|
[*][03/01/2021 09:37] - Identified shares created in last 90 days.
|
|
[*][03/01/2021 09:37] - Identified shares accessed in last 90 days.
|
|
[*][03/01/2021 09:37] - Identified shares modified in last 90 days.
|
|
[*][03/01/2021 09:37] Analysis Complete
|
|
---------------------------------------------------------------
|
|
SHARE REPORT SUMMARY
|
|
---------------------------------------------------------------
|
|
[*][03/01/2021 09:37] Domain: demo.local
|
|
[*][03/01/2021 09:37] Start time: 03/01/2021 09:35:04
|
|
[*][03/01/2021 09:37] End time: 03/01/2021 09:37:27
|
|
[*][03/01/2021 09:37] Run time: 00:02:23.2759086
|
|
[*][03/01/2021 09:37]
|
|
[*][03/01/2021 09:37] COMPUTER SUMMARY
|
|
[*][03/01/2021 09:37] - 245 domain computers found.
|
|
[*][03/01/2021 09:37] - 55 (22.45%) domain computers responded to ping.
|
|
[*][03/01/2021 09:37] - 49 (20.00%) domain computers had TCP port 445 accessible.
|
|
[*][03/01/2021 09:37] - 32 (13.06%) domain computers had shares that were non-default.
|
|
[*][03/01/2021 09:37] - 12 (4.90%) domain computers had shares with potentially excessive privileges.
|
|
[*][03/01/2021 09:37] - 12 (4.90%) domain computers had shares that allowed READ access.
|
|
[*][03/01/2021 09:37] - 1 (0.41%) domain computers had shares that allowed WRITE access.
|
|
[*][03/01/2021 09:37] - 0 (0.00%) domain computers had shares that are HIGH RISK.
|
|
[*][03/01/2021 09:37]
|
|
[*][03/01/2021 09:37] SHARE SUMMARY
|
|
[*][03/01/2021 09:37] - 217 shares were found. We expect a minimum of 98 shares
|
|
[*][03/01/2021 09:37] because 49 systems had open ports and there are typically two default shares.
|
|
[*][03/01/2021 09:37] - 46 (21.20%) shares across 32 systems were non-default.
|
|
[*][03/01/2021 09:37] - 14 (6.45%) shares across 12 systems are configured with 33 potentially excessive ACLs.
|
|
[*][03/01/2021 09:37] - 14 (6.45%) shares across 12 systems allowed READ access.
|
|
[*][03/01/2021 09:37] - 1 (0.46%) shares across 1 systems allowed WRITE access.
|
|
[*][03/01/2021 09:37] - 0 (0.00%) shares across 0 systems are considered HIGH RISK.
|
|
[*][03/01/2021 09:37]
|
|
[*][03/01/2021 09:37] SHARE ACL SUMMARY
|
|
[*][03/01/2021 09:37] - 374 ACLs were found.
|
|
[*][03/01/2021 09:37] - 374 (100.00%) ACLs were associated with non-default shares.
|
|
[*][03/01/2021 09:37] - 33 (8.82%) ACLs were found to be potentially excessive.
|
|
[*][03/01/2021 09:37] - 32 (8.56%) ACLs were found that allowed READ access.
|
|
[*][03/01/2021 09:37] - 1 (0.27%) ACLs were found that allowed WRITE access.
|
|
[*][03/01/2021 09:37] - 0 (0.00%) ACLs were found that are associated with HIGH RISK share names.
|
|
[*][03/01/2021 09:37]
|
|
[*][03/01/2021 09:37] - The 5 most common share names are:
|
|
[*][03/01/2021 09:37] - 9 of 14 (64.29%) discovered shares are associated with the top 5 share names.
|
|
[*][03/01/2021 09:37] - 4 backup
|
|
[*][03/01/2021 09:37] - 2 ssms
|
|
[*][03/01/2021 09:37] - 1 test2
|
|
[*][03/01/2021 09:37] - 1 test1
|
|
[*][03/01/2021 09:37] - 1 users
|
|
[*] -----------------------------------------------
|
|
|
|
#>
|
|
[CmdletBinding()]
|
|
Param(
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Domain user to authenticate with domain\user. For computer lookup.')]
|
|
[string]$Username,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Domain password to authenticate with domain\user. For computer lookup.')]
|
|
[string]$Password,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Credentials to use when connecting to a Domain Controller. For computer lookup.')]
|
|
[System.Management.Automation.PSCredential]
|
|
[System.Management.Automation.Credential()]$Credential = [System.Management.Automation.PSCredential]::Empty,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Domain controller for Domain and Site that you want to query against. For computer lookup.')]
|
|
[string]$DomainController,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Number of threads to process at once.')]
|
|
[int]$Threads = 20,
|
|
|
|
[Parameter(Mandatory = $true,
|
|
HelpMessage = 'Directory to output files to.')]
|
|
[string]$OutputDirectory,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Creat exported csv for import into other tools.')]
|
|
[switch]$ExportFindings,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'This is the path to a host list. One per line.')]
|
|
[string] $HostList,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Number of items to sample for summary report.')]
|
|
[int]$SampleSum = 5,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Runspace time out.')]
|
|
[int]$RunSpaceTimeOut = 15,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Time in days for recent share creation summary report.')]
|
|
[int]$ShareCreationDays = 90,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Time in days for last access report.')]
|
|
[int]$LastAccessDays = 90,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Time in days for last modified report.')]
|
|
[int]$LastModDays = 90,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Show runspace errors if they occur.')]
|
|
[switch] $ShowRunpaceErrors,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Supress timeline report generation.')]
|
|
[switch] $SupressTimelineRpt,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Number of directory levels to capture file list.')]
|
|
[int] $DirLevel = 3,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Search for common files containing passwords on the c$ shares.')]
|
|
[switch] $FindFiles,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Path to file of file paths to search for. One path per line.')]
|
|
[string] $FindFilesList,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Do not perform ping scan.')]
|
|
[switch] $NoPing
|
|
|
|
)
|
|
|
|
|
|
Begin
|
|
{
|
|
Write-Output " ==============================================================="
|
|
Write-Output " INVOKE-HUNTSMBSHARES "
|
|
Write-Output " ==============================================================="
|
|
Write-Output " This function automates the following tasks: "
|
|
Write-Output " "
|
|
Write-Output " o Determine current computer's domain "
|
|
Write-Output " o Enumerate domain computers "
|
|
Write-Output " o Check if computers respond to ping requests "
|
|
Write-Output " o Filter for computers that have TCP 445 open and accessible "
|
|
Write-Output " o Enumerate SMB shares "
|
|
Write-Output " o Enumerate SMB share permissions "
|
|
Write-Output " o Identify shares with potentially excessive privielges "
|
|
Write-Output " o Identify shares that provide read or write access "
|
|
Write-Output " o Identify shares thare are high risk "
|
|
Write-Output " o Identify common share owners, names, & directory listings "
|
|
Write-Output " o Generate last written & last accessed timelines "
|
|
Write-Output " o Generate html summary report and detailed csv files "
|
|
Write-Output ""
|
|
Write-Output " Note: This can take hours to run in large environments. "
|
|
Write-Output " ---------------------------------------------------------------"
|
|
Write-Output " |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||"
|
|
Write-Output " ---------------------------------------------------------------"
|
|
Write-Output " SHARE DISCOVERY "
|
|
Write-Output " ---------------------------------------------------------------"
|
|
|
|
|
|
# Get start time
|
|
$StartTime = Get-Date
|
|
$StopWatch = [system.diagnostics.stopwatch]::StartNew()
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Scan Start"
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Create output directory
|
|
# ----------------------------------------------------------------------
|
|
if(Test-Path $OutputDirectory){
|
|
|
|
# Verify output directory path
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
$FolderDateTime = Get-Date -Format "MMddyyyyHHmmss"
|
|
$OutputDirectoryBase = "$OutputDirectory\SmbShareHunt-$FolderDateTime"
|
|
|
|
# Create sub directories
|
|
mkdir $OutputDirectoryBase | Out-Null
|
|
$SubDir = "Results"
|
|
mkdir "$OutputDirectoryBase\$SubDir" | Out-Null
|
|
$OutputDirectory = "$OutputDirectoryBase\Results"
|
|
Write-Output " [*][$Time] Output Directory: $OutputDirectoryBase"
|
|
}else{
|
|
Write-Output " [x][$Time] The $OutputDirectory was not writable."
|
|
Write-Output " [!][$Time] Aborting operation."
|
|
break
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Import computers from file
|
|
# ----------------------------------------------------------------------
|
|
if($HostList){
|
|
|
|
if(test-path $HostList){
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Importing computer targets from $HostList"
|
|
$HostListContent = gc $HostList
|
|
$DomainComputers = $HostListContent |
|
|
foreach {
|
|
$object = New-Object psobject
|
|
$Object | Add-Member Noteproperty ComputerName $_
|
|
$Object
|
|
}
|
|
$TargetDomain = "SmbHunt"
|
|
$ComputerCount = $DomainComputers | measure | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] $ComputerCount systems will be targeted"
|
|
}else{
|
|
Write-Output " [!][$Time] The host list was not accessible: $HostList"
|
|
break
|
|
}
|
|
}
|
|
|
|
# Set variables
|
|
$GlobalThreadCount = $Threads
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Enumerate domain computers
|
|
# ----------------------------------------------------------------------
|
|
|
|
if(-not $HostList){
|
|
|
|
# Create PS Credential object
|
|
if($Username -and $Password)
|
|
{
|
|
$secpass = ConvertTo-SecureString $Password -AsPlainText -Force
|
|
$Credential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList ($Username, $secpass)
|
|
}
|
|
|
|
# Set target domain
|
|
$DCRecord = Get-LdapQuery -LdapFilter "(&(objectCategory=computer)(userAccountControl:1.2.840.113556.1.4.803:=8192))" -DomainController $DomainController -Credential $Credential | select -first 1 | select properties -expand properties -ErrorAction SilentlyContinue
|
|
[string]$DCHostname = $DCRecord.dnshostname
|
|
[string]$DCCn = $DCRecord.cn
|
|
[string]$TargetDomain = $DCHostname -replace ("$DCCn\.","")
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
if($DCHostname)
|
|
{
|
|
Write-Output " [*][$Time] Successful connection to domain controller: $DCHostname"
|
|
}else{
|
|
Write-Output " [*][$Time] There appears to have been an error connecting to the domain controller."
|
|
Write-Output " [*][$Time] Aborting."
|
|
break
|
|
}
|
|
|
|
# Status user
|
|
Write-Output " [*][$Time] Performing LDAP query for computers associated with the $TargetDomain domain"
|
|
|
|
# Get domain computers
|
|
$DomainComputersRecord = Get-LdapQuery -LdapFilter "(objectCategory=Computer)" -DomainController $DomainController -Credential $Credential
|
|
$DomainComputers = $DomainComputersRecord |
|
|
foreach{
|
|
|
|
$DnsHostName = [string]$_.Properties['dnshostname']
|
|
if($DnsHostName -notlike ""){
|
|
$object = New-Object psobject
|
|
$Object | Add-Member Noteproperty ComputerName $DnsHostName
|
|
$Object
|
|
}
|
|
}
|
|
|
|
# Status user
|
|
$ComputerCount = $DomainComputers.count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $ComputerCount computers found"
|
|
|
|
# Save results
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Domain-Computers.csv"
|
|
$DomainComputers | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Domain-Computers.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $DomainComputers -Outfile "$OutputDirectory\$TargetDomain-Domain-Computers.html" -Title "Domain Computers" -Description "This page shows the domain computers discovered for the $TargetDomain Active Directory domain."
|
|
$DomainComputersFile = "$TargetDomain-Domain-Computers.csv"
|
|
$DomainComputersFileH = "$TargetDomain-Domain-Computers.html"
|
|
|
|
# Get subnet info
|
|
$DomainSubnets = Get-DomainSubnet -DomainController $DomainController -Credential $Credential
|
|
$DomainSubnetsCount = $DomainSubnets | measure | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $DomainSubnetsCount subnets found"
|
|
$DomainSubnets | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Domain-Subnets.csv"
|
|
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify computers that respond to ping reqeusts
|
|
# ----------------------------------------------------------------------
|
|
|
|
if($NoPing){
|
|
Write-Output " [*][$Time] - Skipping ping scan."
|
|
$ComputerPingableCount = 0
|
|
}else{
|
|
|
|
# Status user
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Pinging $ComputerCount computers"
|
|
|
|
# Ping computerss
|
|
$PingResults = $DomainComputers | Invoke-Ping -Throttle $GlobalThreadCount
|
|
|
|
# select computers that respond
|
|
$ComputersPingable = $PingResults |
|
|
foreach {
|
|
|
|
$computername = $_.address
|
|
$status = $_.status
|
|
if($status -like "Responding"){
|
|
$object = new-object psobject
|
|
$Object | add-member Noteproperty ComputerName $computername
|
|
$Object | add-member Noteproperty status $status
|
|
$Object
|
|
}
|
|
}
|
|
|
|
# Status user
|
|
$ComputerPingableCount = $ComputersPingable.count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $ComputerPingableCount computers responded to ping requests."
|
|
|
|
# Save results
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Domain-Computers-Pingable.csv"
|
|
if($ComputersPingable){
|
|
$ComputersPingable | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Domain-Computers-Pingable.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $ComputersPingable -Outfile "$OutputDirectory\$TargetDomain-Domain-Computers-Pingable.html" -Title "Domain Computers: Ping Response" -Description "This page shows the domain computers for the $TargetDomain Active Directory domain that responded to ping requests."
|
|
}
|
|
$ComputersPingableFile = "$TargetDomain-Domain-Computers-Pingable.csv"
|
|
$ComputersPingableFileH = "$TargetDomain-Domain-Computers-Pingable.html"
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify computers that have TCP 445 open and accessible
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Status user
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Checking if TCP Port 445 is open on $ComputerCount computers"
|
|
|
|
# Create script block to port scan tcp 445
|
|
$MyScriptBlock = {
|
|
$ComputerName = $_.ComputerName
|
|
try{
|
|
$Socket = New-Object System.Net.Sockets.TcpClient($ComputerName,"445")
|
|
|
|
if($Socket.Connected)
|
|
{
|
|
$Status = "Open"
|
|
$Socket.Close()
|
|
}
|
|
else
|
|
{
|
|
$Status = "Closed"
|
|
}
|
|
}
|
|
catch{
|
|
$Status = "Closed"
|
|
}
|
|
|
|
if($Status -eq "Open")
|
|
{
|
|
$object = new-object psobject
|
|
$Object | add-member Noteproperty ComputerName $computername
|
|
$Object | add-member Noteproperty 445status $status
|
|
$Object
|
|
}
|
|
}
|
|
|
|
# Perform port scan of tcp 445 threaded
|
|
$Computers445Open = $DomainComputers | Invoke-Parallel -ScriptBlock $MyScriptBlock -ImportSessionFunctions -ImportVariables -Throttle $GlobalThreadCount -RunspaceTimeout $RunSpaceTimeOut -ErrorAction SilentlyContinue
|
|
|
|
# Status user
|
|
$Computers445OpenCount = $Computers445Open.count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $Computers445OpenCount computers have TCP port 445 open."
|
|
|
|
|
|
# Stop if no ports are accessible
|
|
If ($Computers445OpenCount -eq 0)
|
|
{
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Aborting."
|
|
break
|
|
}
|
|
|
|
# Save results
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Domain-Computers-Open445.csv"
|
|
if($Computers445Open){
|
|
$Computers445Open | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Domain-Computers-Open445.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $Computers445Open -Outfile "$OutputDirectory\$TargetDomain-Domain-Computers-Open445.html" -Title "Domain Computers: Port 445 Open" -Description "This page shows the domain computers for the $TargetDomain Active Directory domain with port 445 open."
|
|
}
|
|
$Computers445OpenFile = "$TargetDomain-Domain-Computers-Open445.csv"
|
|
$Computers445OpenFileH ="$TargetDomain-Domain-Computers-Open445.html"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Enumerate computer SMB shares
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Status user
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Getting a list of SMB shares from $Computers445OpenCount computers"
|
|
|
|
# Create script block to query for SMB shares
|
|
$MyScriptBlock = {
|
|
Get-MySMBShare -ComputerName $_.ComputerName
|
|
}
|
|
|
|
# Get smb shares threaded
|
|
$AllSMBShares = $Computers445Open | Invoke-Parallel -ScriptBlock $MyScriptBlock -ImportSessionFunctions -ImportVariables -Throttle $GlobalThreadCount -RunspaceTimeout $RunSpaceTimeOut -ErrorAction SilentlyContinue
|
|
|
|
# Computer computers with shares
|
|
$AllComputersWithShares = $AllSMBShares | Select-Object ComputerName -Unique
|
|
$AllComputersWithSharesCount = $AllComputersWithShares.count
|
|
|
|
# Status user
|
|
$AllSMBSharesCount = $AllSMBShares.count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $AllSMBSharesCount SMB shares were found."
|
|
|
|
# Stop if no shares
|
|
If ($AllSMBSharesCount -eq 0)
|
|
{
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Aborting."
|
|
break
|
|
}
|
|
|
|
# Save results
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-All.csv"
|
|
$AllSMBShares | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-All.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $AllSMBShares -Outfile "$OutputDirectory\$TargetDomain-Shares-Inventory-All.html" -Title "Domain Shares" -Description "This page shows the all enumerated shares for the $TargetDomain Active Directory domain."
|
|
$AllSMBSharesFile = "$TargetDomain-Shares-Inventory-All.csv"
|
|
$AllSMBSharesFileH = "$TargetDomain-Shares-Inventory-All.html"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Enumerate computer SMB share permissions
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Status user
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Getting share permissions from $AllSMBSharesCount SMB shares"
|
|
|
|
# Create script block to query for SMB permissions
|
|
$MyScriptBlock = {
|
|
|
|
$CurrentShareName = $_.ShareName
|
|
$CurrentComputerName = $_.ComputerName
|
|
$CurrentIP = $_.IpAddress
|
|
$ShareDescription = $_.ShareDesc
|
|
$Sharetype = $_.sharetype
|
|
$Shareaccess = $_.shareaccess
|
|
|
|
if($CurrentComputerName -eq ""){
|
|
$TargetAsset = $CurrentIP
|
|
}else{
|
|
$TargetAsset = $CurrentComputerName
|
|
}
|
|
|
|
$currentaacl = Get-PathAcl "\\$TargetAsset\$CurrentShareName" -ErrorAction SilentlyContinue
|
|
$currentaacl |
|
|
foreach{
|
|
|
|
# Get file listing
|
|
$FullFileList = Get-ChildItem -Path "\\$TargetAsset\$CurrentShareName"
|
|
|
|
# Get file count
|
|
$FileCount = $FullFileList.Count
|
|
|
|
# Get top 5 files list
|
|
$FileList = $FullFileList | Select-Object Name -ExpandProperty Name | Out-String
|
|
|
|
# Get File listing hash
|
|
$FileListGroup = Get-FolderGroupMd5 -FolderList $FileList
|
|
$FileListGroup
|
|
|
|
# Audit Settings
|
|
$AuditSettings = Get-ACL "\\$TargetAsset\$CurrentShareName" -ErrorAction SilentlyContinue | Select-Object AuditToString -ExpandProperty AuditToString
|
|
|
|
# Creation date
|
|
$CreationdDateObject = Get-Item "\\$TargetAsset\$CurrentShareName" -ErrorAction SilentlyContinue | Select CreationTime
|
|
$CreationdDate = $CreationdDateObject.CreationTime.ToString()
|
|
$CreationdDateYear = $CreationdDateObject.CreationTime.Year.ToString()
|
|
|
|
# Last modified date
|
|
$TargetPath = $_.Path
|
|
$LastModifiedDateObject = Get-Item "\\$TargetAsset\$CurrentShareName" -ErrorAction SilentlyContinue | Select-Object LastWriteTime
|
|
$LastModifiedDate = $LastModifiedDateObject.LastWriteTime.ToString()
|
|
$LastModifiedDateYear = $LastModifiedDate.split(' ')[0].split('/')[2]
|
|
|
|
# Last accessed date
|
|
$TargetPath = $_.Path
|
|
$LastAccessDateObject = Get-Item "\\$TargetAsset\$CurrentShareName" -ErrorAction SilentlyContinue | Select-Object LastAccessTime
|
|
$LastAccessDate = $LastAccessDateObject.LastAccessTime.ToString()
|
|
$LastAccessDateYear = $LastAccessDate.split(' ')[0].split('/')[2]
|
|
|
|
$aclObject = new-object psobject
|
|
$aclObject | add-member Noteproperty ComputerName $CurrentComputerName
|
|
$aclObject | add-member Noteproperty IpAddress $CurrentIP
|
|
$aclObject | add-member Noteproperty ShareName $CurrentShareName
|
|
$aclObject | add-member Noteproperty SharePath $_.Path
|
|
$aclObject | add-member Noteproperty ShareDescription $ShareDescription
|
|
$aclObject | add-member Noteproperty ShareOwner $_.PathOwner
|
|
$aclObject | add-member Noteproperty ShareType $ShareType
|
|
$aclObject | add-member Noteproperty ShareAccess $ShareAccess
|
|
$aclObject | add-member Noteproperty FileSystemRights $_.FileSystemRights
|
|
$aclObject | add-member Noteproperty IdentityReference $_.IdentityReference
|
|
$aclObject | add-member Noteproperty IdentitySID $_.IdentitySID
|
|
$aclObject | add-member Noteproperty AccessControlType $_.AccessControlType
|
|
$aclObject | add-member Noteproperty CreationDate $CreationdDate
|
|
$aclObject | add-member Noteproperty CreationDateYear $CreationdDateYear
|
|
$aclObject | add-member Noteproperty LastModifiedDate $LastModifiedDate
|
|
$aclObject | add-member Noteproperty LastModifiedDateYear $LastModifiedDateYear
|
|
$aclObject | add-member Noteproperty LastAccessDate $LastAccessDate
|
|
$aclObject | add-member Noteproperty LastAccessDateYear $LastAccessDateYear
|
|
$aclObject | add-member Noteproperty FileCount $FileCount
|
|
$aclObject | add-member Noteproperty FileList $FileList
|
|
$aclObject | add-member Noteproperty FileListGroup $FileListGroup
|
|
$aclObject | add-member Noteproperty AuditSettings $AuditSettings
|
|
$aclObject
|
|
}
|
|
}
|
|
|
|
# Get SMB permissions threaded
|
|
$ShareACLs = $AllSMBShares | Invoke-Parallel -ScriptBlock $MyScriptBlock -ImportSessionFunctions -ImportVariables -Throttle $GlobalThreadCount -RunspaceTimeout $RunSpaceTimeOut -ErrorAction SilentlyContinue -WarningAction SilentlyContinue | where ShareName -notlike ""
|
|
|
|
# Status user
|
|
$ShareACLsCount = $ShareACLs | measure | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $ShareACLsCount share permissions were enumerated."
|
|
|
|
# Stop if no shares ACLs were enumerated
|
|
If ($ShareACLsCount -eq 0)
|
|
{
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Aborting."
|
|
break
|
|
}
|
|
|
|
# Save results
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-All-ACL.csv"
|
|
$ShareACLs | where ShareName -notlike "" | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-All-ACL.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $ShareACLs -Outfile "$OutputDirectory\$TargetDomain-Shares-Inventory-All-ACL.html" -Title "Domain Shares: All ACL Entries" -Description "This page shows all share ACL entries discovered on computers associated with the $TargetDomain Active Directory domain."
|
|
$ShareACLsFile = "$TargetDomain-Shares-Inventory-All-ACL.csv"
|
|
$ShareACLsFileH = "$TargetDomain-Shares-Inventory-All-ACL.html"
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Get potentially excessive share permissions
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Status user
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Identifying potentially excessive share permissions"
|
|
|
|
# Check for share that provide read/write access to common user groups
|
|
$ExcessiveSharePrivs = foreach ($line in $ShareACLs){
|
|
|
|
# Filter for basic user ACLs
|
|
if (($line.IdentityReference -eq "Everyone") -or ($line.IdentityReference -eq "BUILTIN\Users") -or ($line.IdentityReference -eq "Authenticated Users") -or ($line.IdentityReference -like "*Domain Users*") ){
|
|
|
|
if($line.ShareAccess -like "Yes"){
|
|
|
|
if(($line.ShareName -notlike "print$") -and ($line.ShareName -notlike "prnproc$") -and ($line.ShareName -notlike "*printer*") -and ($line.ShareName -notlike "sysvol") -and ($line.ShareName -notlike "netlogon"))
|
|
{
|
|
$line
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Status user
|
|
$ExcessiveAclCount = $ExcessiveSharePrivs.count
|
|
$ExcessiveShares = $ExcessiveSharePrivs | Select-Object ComputerName,ShareName -unique
|
|
$ExcessiveSharesCount = $ExcessiveShares.count
|
|
$ExcessiveSharePrivsCount = $ExcessiveSharePrivs.count
|
|
$ComputerWithExcessive = $ExcessiveSharePrivs | Select-Object ComputerName -Unique | Measure-Object | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $ExcessiveSharePrivsCount potentially excessive privileges were found on $ExcessiveSharesCount shares across $ComputerWithExcessive systems."
|
|
|
|
# Save results
|
|
if($ExcessiveSharesCount -ne 0){
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges.csv"
|
|
$ExcessiveSharePrivs | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $ExcessiveSharePrivs -Outfile "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges.html" -Title "Domain Shares: ACL Entries - Excessive Privileges" -Description "This page shows all share ACL entries discovered on computers associated with the $TargetDomain Active Directory domain that appear to be configured with excessive privileges."
|
|
$ShareACLsExFile = "$TargetDomain-Shares-Inventory-Excessive-Privileges.csv"
|
|
$ShareACLsExFileH = "$TargetDomain-Shares-Inventory-Excessive-Privileges.html"
|
|
}else{
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [!][$Time] 0 excessive privileges found."
|
|
break
|
|
}
|
|
|
|
$ExcessiveSharePrivsFile = "$TargetDomain-Shares-Inventory-Excessive-Privileges.csv"
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Get Recursive Directory Listings
|
|
# ----------------------------------------------------------------------
|
|
# Default depth is 3 by default
|
|
|
|
# Status user
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Getting directory listings from $ExcessiveSharesCount SMB shares"
|
|
Write-Output " [*][$Time] - Targeting up to $DirLevel nested directory levels"
|
|
|
|
# Create script block to query for directory listing
|
|
$MyScriptBlock = {
|
|
|
|
# Get share context
|
|
$CurrentComputerName = $_.ComputerName
|
|
$CurrentIP = $_.IpAddress
|
|
$ShareDescription = $_.ShareDesc
|
|
$CurrentShareName = $_.ShareName
|
|
$SharePath = $_.SharePath
|
|
|
|
# Get file listing from non system directories
|
|
#$FullFileList = Get-ChildItem -Path $SharePath -Exclude "c:\windows" | Select FullName
|
|
$FullFileList = Get-ChildItem -Depth $DirLevel -Path "$SharePath" | select fullname | where {$_.fullname -NotLike "$SharePath\Windows*" -and $_.fullname -notlike "$SharePath\WINNT"}
|
|
|
|
$FullFileList |
|
|
Foreach{
|
|
$aclObject = new-object psobject
|
|
$aclObject | add-member Noteproperty ComputerName $CurrentComputerName
|
|
$aclObject | add-member Noteproperty IpAddress $CurrentIP
|
|
$aclObject | add-member Noteproperty ShareName $CurrentShareName
|
|
$aclObject | add-member Noteproperty SharePath $SharePath
|
|
$aclObject | add-member Noteproperty ShareDescription $ShareDescription
|
|
$aclObject | add-member Noteproperty FilePath $_.fullname
|
|
$aclObject
|
|
}
|
|
}
|
|
|
|
# Get SMB directory listing threaded
|
|
$ShareDirListing = $ExcessiveSharePrivs | select computername,ipaddress,sharedesc,sharename,sharepath -Unique | Invoke-Parallel -ScriptBlock $MyScriptBlock -ImportSessionFunctions -ImportVariables -Throttle $GlobalThreadCount -RunspaceTimeout $RunSpaceTimeOut -ErrorAction SilentlyContinue -WarningAction SilentlyContinue | where ShareName -notlike "" #| select * -Unique
|
|
|
|
# Status user
|
|
$ShareDirListingCount = $ShareDirListing | measure | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $ShareDirListingCount files and folders were enumerated."
|
|
|
|
# Write output
|
|
# Write-Output " [*] Saving results to $OutputDirectory\$TargetDomain-Shares-Directory-Listings-Depth-$DirLevel.csv"
|
|
$ShareDirListing | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Directory-Listings-Depth-$DirLevel.csv"
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Get File Listings from Shares
|
|
# ----------------------------------------------------------------------
|
|
#$FindFiles # hardcoded array
|
|
#$FindFilesList # file
|
|
# combine
|
|
# create where statement
|
|
#$ShareDirListing | Where
|
|
#count
|
|
#export
|
|
|
|
# End of scanning
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Scan Complete"
|
|
|
|
Write-Output " ---------------------------------------------------------------"
|
|
Write-Output " SHARE ANALYSIS "
|
|
Write-Output " ---------------------------------------------------------------"
|
|
Write-Output " [*][$Time] Analysis Start"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify shares that provide read access
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Get shares that provide read access
|
|
$SharesWithread = $ExcessiveSharePrivs |
|
|
Foreach {
|
|
|
|
if(($_.FileSystemRights -like "*read*"))
|
|
{
|
|
$_ # out to file
|
|
}
|
|
}
|
|
|
|
# Status user
|
|
$AclWithReadCount = $SharesWithread.count
|
|
$SharesWithReadCount = $SharesWithread | Select-Object SharePath -Unique | Measure-Object | select count -ExpandProperty count
|
|
$ComputerWithReadCount = $SharesWithread | Select-Object ComputerName -Unique | Measure-Object | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $SharesWithReadCount shares can be read across $ComputerWithReadCount systems."
|
|
|
|
# Save results
|
|
if($SharesWithReadCount -ne 0){
|
|
#Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Read.csv"
|
|
$SharesWithRead | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Read.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $SharesWithRead -Outfile "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Read.html" -Title "Domain Shares: ACL Allow Read Entries" -Description "This page shows all share ACL entries discovered on computers associated with the $TargetDomain Active Directory domain that are readable."
|
|
$ShareACLsReadFile = "$TargetDomain-Shares-Inventory-Excessive-Privileges-Read.csv"
|
|
$ShareACLsReadFileH = "$TargetDomain-Shares-Inventory-Excessive-Privileges-Read.html"
|
|
}
|
|
|
|
$SharesWithReadFile = "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Read.csv"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify shares that provide write access
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Get shares that provide write access
|
|
$SharesWithWrite = $ExcessiveSharePrivs |
|
|
Foreach {
|
|
|
|
if(($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*"))
|
|
{
|
|
$_ # out to file
|
|
}
|
|
}
|
|
|
|
# Status user
|
|
$AclWithWriteCount = $SharesWithWrite | Measure-Object | select count -ExpandProperty count
|
|
$SharesWithWriteCount = $SharesWithWrite | Select-Object SharePath -Unique | Measure-Object | select count -ExpandProperty count
|
|
$ComputerWithWriteCount = $SharesWithWrite | Select-Object ComputerName -Unique | Measure-Object | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $SharesWithWriteCount shares can be written to across $ComputerWithWriteCount systems."
|
|
|
|
# Save results
|
|
if($SharesWithWriteCount -ne 0){
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Write.csv"
|
|
$SharesWithWrite | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Write.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $SharesWithWrite -Outfile "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Write.html" -Title "Domain Shares: ACL Allow Write Entries" -Description "This page shows all share ACL entries discovered on computers associated with the $TargetDomain Active Directory domain that are writable."
|
|
$ShareACLsWriteFile = "$TargetDomain-Shares-Inventory-Excessive-Privileges-Write.csv"
|
|
$ShareACLsWriteFileH = "$TargetDomain-Shares-Inventory-Excessive-Privileges-Write.html"
|
|
}
|
|
|
|
$SharesWithWriteFile = "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-Write.csv"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify shares that are non-default
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Get high risk share access
|
|
$SharesNonDefault = $ShareACLs |
|
|
Foreach {
|
|
|
|
if(($_.ShareName -notlike 'admin$') -or ($_.ShareName -notlike 'c$') -or ($_.ShareName -notlike 'd$') -or ($_.ShareName -notlike 'e$') -or ($_.ShareName -notlike 'f$'))
|
|
{
|
|
$_ # out to file
|
|
}
|
|
}
|
|
|
|
# Status user
|
|
$AclNonDefaultCount = $SharesNonDefault | measure | select count -ExpandProperty count
|
|
$SharesNonDefaultCount = $SharesNonDefault | Select-Object SharePath -Unique | Measure-Object | select count -ExpandProperty count
|
|
$ComputerwithNonDefaultCount = $SharesNonDefault | Select-Object ComputerName -Unique | Measure-Object | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $SharesNonDefaultCount shares are considered non-default across $ComputerwithNonDefaultCount systems."
|
|
|
|
# Save results
|
|
if($SharesNonDefaultCount-ne 0){
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-NonDefault.csv"
|
|
$SharesNonDefault | where ShareName -notlike "" | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-NonDefault.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $SharesNonDefault -Outfile "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-NonDefault.html" -Title "Domain Shares: Non-Default" -Description "This page shows all share ACL entries discovered on computers associated with the $TargetDomain Active Directory domain that are non-default."
|
|
$ShareACLsNonDefaultFile = "$TargetDomain-Shares-Inventory-Excessive-Privileges-NonDefault.csv"
|
|
$ShareACLsNonDefaultFileH = "$TargetDomain-Shares-Inventory-Excessive-Privileges-NonDefault.html"
|
|
}
|
|
|
|
$SharesNonDefaultFile = "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-NonDefault.csv"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify shares that are high risk
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Get high risk share access
|
|
$SharesHighRisk = $ExcessiveSharePrivs |
|
|
Foreach {
|
|
|
|
if(($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share'))
|
|
{
|
|
$_ # out to file
|
|
}
|
|
}
|
|
|
|
# Status user
|
|
$AclHighRiskCount = $SharesHighRisk.count
|
|
$SharesHighRiskCount = $SharesHighRisk | Select-Object SharePath -Unique | Measure-Object | select count -ExpandProperty count
|
|
$ComputerwithHighRisk = $SharesHighRisk | Select-Object ComputerName -Unique | Measure-Object | select count -ExpandProperty count
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - $SharesHighRiskCount shares are considered high risk across $ComputerwithHighRisk systems."
|
|
|
|
# Save results
|
|
if($SharesHighRiskCount -ne 0){
|
|
# Write-Output " [*] - Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-HighRisk.csv"
|
|
$SharesHighRisk | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-HighRisk.csv"
|
|
$null = Convert-DataTableToHtmlTable -DataTable $SharesHighRisk -Outfile "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-HighRisk.html" -Title "Domain Shares: ACL High Risk Entries" -Description "This page shows all share ACL entries discovered on computers associated with the $TargetDomain Active Directory domain that are considered to be high risk."
|
|
$ShareACLsHRFile = "$TargetDomain-Shares-Inventory-Excessive-Privileges-HighRisk.csv"
|
|
$ShareACLsHRFileH = "$TargetDomain-Shares-Inventory-Excessive-Privileges-HighRisk.html"
|
|
}
|
|
|
|
$SharesHighRiskFile = "$OutputDirectory\$TargetDomain-Shares-Inventory-Excessive-Privileges-HighRisk.csv"
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify common excessive share owners
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Get share owner list
|
|
$CommonShareOwners = $ExcessiveSharePrivs | Select SharePath,ShareOwner -Unique |
|
|
Select-Object ShareOwner |
|
|
<#
|
|
where ShareOwner -notlike "BUILTIN\Administrators" |
|
|
where ShareOwner -notlike "NT AUTHORITY\SYSTEM" |
|
|
where ShareOwner -notlike "NT SERVICE\TrustedInstaller" |
|
|
#>
|
|
Group-Object ShareOwner |
|
|
Sort-Object ShareOwner |
|
|
Select-Object count,name |
|
|
Sort-Object Count -Descending
|
|
|
|
# Save list
|
|
$CommonShareOwners | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Common-Owners.csv"
|
|
$CommonShareOwnersCount = $CommonShareOwners | measure | select count -ExpandProperty count
|
|
|
|
# Get top 5
|
|
$CommonShareOwnersTop5 = $CommonShareOwners | Select-Object count,name -First $SampleSum
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Identified top 5 owners of excessive shares."
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify common excessive share groups (group by file list)
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Get share owner list
|
|
$CommonShareFileGroup = $ExcessiveSharePrivs |
|
|
Select-Object FileListGroup |
|
|
Group-Object FileListGroup|
|
|
Select-Object count,name |
|
|
Sort-Object Count -Descending
|
|
|
|
# Save list
|
|
$CommonShareFileGroup | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Common-FileGroups.csv"
|
|
$CommonShareFileGroupCount = $CommonShareFileGroup.count
|
|
|
|
# Get top 5
|
|
$CommonShareFileGroupTop5 = $CommonShareFileGroup | Select-Object count,name,filecount -First $SampleSum
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Identified top 5 share groups."
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify common share names
|
|
# ----------------------------------------------------------------------
|
|
|
|
$CommonShareNames = $ExcessiveSharePrivs | Select-Object ComputerName,ShareName -Unique | Group-Object ShareName |Sort Count -Descending | select count,name |
|
|
foreach{
|
|
if( ($_.name -ne 'SYSVOL') -and ($_.name -ne 'NETLOGON'))
|
|
{
|
|
$_
|
|
}
|
|
}
|
|
|
|
# Get percent of shared covered by top 5
|
|
# If very weighted this indicates if the shares are part of a deployment process, image, or app
|
|
|
|
# Get top five share name
|
|
$CommonShareNamesCount = $CommonShareNames.count
|
|
$CommonShareNamesTop5 = $CommonShareNames | Select-Object count,name -First $SampleSum
|
|
|
|
# Get count of share name if in the top 5
|
|
$Top5ShareCountTotal = 0
|
|
$CommonShareNamesTop5 |
|
|
foreach{
|
|
[int]$TopCount = $_.Count
|
|
$Top5ShareCountTotal = $Top5ShareCountTotal + $TopCount
|
|
}
|
|
|
|
# Get count of all accessible shares
|
|
$AllAccessibleSharesCount = $ExcessiveSharePrivs | Select-Object ComputerName,ShareName -Unique | measure | select count -ExpandProperty count
|
|
|
|
# Write output
|
|
#Write-Output " [*] Saving results to $OutputDirectory\$TargetDomain-Shares-Inventory-Common-Names.csv"
|
|
$CommonShareNames | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Common-Names.csv"
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Identified top 5 share names."
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify excessive share creation in last n 90 days
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Select shares from last n names
|
|
$StartDateAccess = (get-date).AddDays(-$ShareCreationDays);
|
|
$EndDateAccess = Get-Date
|
|
$ExPrivCreationLastn = $ExcessiveSharePrivs | Where-Object {([Datetime]$_.CreationDate.trim() -ge $StartDateAccess -and [Datetime]$_.CreationDate.trim() -le $EndDateAccess)}
|
|
$ExPrivCreationLastnShare = $ExPrivCreationLastn | select SharePath -Unique
|
|
$ExPrivCreationLastnShareCount = $ExPrivCreationLastnShare | Measure | select count -ExpandProperty count
|
|
|
|
# Percent of shares accessed in last n days
|
|
$ExpPrivCreationLast = $ExPrivCreationLastnShareCount / $AllSMBSharesCount
|
|
$ExpPrivCreationLastP = $ExpPrivCreationLast.tostring("P") -replace(" ","")
|
|
|
|
# Get summary bar code - Need to extend for counts,%, and bar
|
|
$ExPrivCreationLastBars = Get-ExPrivSumData -DataTable $ExPrivCreationLastn -AllComputerCount $ComputerCount -AllShareCount $AllSMBSharesCount -AllAclCount $ShareACLsCount
|
|
$ExPrivCreationLastComputerB = $ExPrivCreationLastBars.ComputerBar
|
|
$ExPrivCreationLastShareB = $ExPrivCreationLastBars.ShareBar
|
|
$ExPrivCreationLastShareAclB = $ExPrivCreationLastBars.AclBar
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Identified shares created in last $ShareCreationDays days."
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify excessive share access in last n 90 days
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Select shares from last n names
|
|
$StartDateAccess = (get-date).AddDays(-$LastAccessDays);
|
|
$EndDateAccess = Get-Date
|
|
$ExPrivAccessLastn = $ExcessiveSharePrivs | Where-Object {([Datetime]$_.LastAccessDate.trim() -ge $StartDateAccess -and [Datetime]$_.LastAccessDate.trim() -le $EndDateAccess)}
|
|
$ExPrivAccessLastnShare = $ExPrivAccessLastn | select SharePath -Unique
|
|
$ExPrivAccessLastnShareCount = $ExPrivAccessLastnShare | Measure | select count -ExpandProperty count
|
|
|
|
# Percent of shares accessed in last n days
|
|
$ExpPrivAccessLast = $ExPrivAccessLastnShareCount / $AllSMBSharesCount
|
|
$ExpPrivAccessLastP = $ExpPrivAccessLast.tostring("P") -replace(" ","")
|
|
|
|
# Get summary bar code - Need to extend for counts,%, and bar
|
|
$ExPrivAccesLastBars = Get-ExPrivSumData -DataTable $ExPrivAccessLastn -AllComputerCount $ComputerCount -AllShareCount $AllSMBSharesCount -AllAclCount $ShareACLsCount
|
|
$ExPrivAccesLastComputerB = $ExPrivAccesLastBars.ComputerBar
|
|
$ExPrivAccesLastShareB = $ExPrivAccesLastBars.ShareBar
|
|
$ExPrivAccesLastShareAclB = $ExPrivAccesLastBars.AclBar
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Identified shares accessed in last $LastAccessDays days."
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify excessive modification in last n 90 days
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Select shares from last n names
|
|
$StartDateModified = (get-date).AddDays(-$LastModDays);
|
|
$EndDateModified = Get-Date
|
|
$ExPrivModifiedLastn = $ExcessiveSharePrivs | Where-Object {([Datetime]$_.LastModifiedDate.trim() -ge $StartDateModified -and [Datetime]$_.LastModifiedDate.trim() -le $EndDateModified)}
|
|
$ExPrivModifiedLastnShare = $ExPrivModifiedLastn | select SharePath -Unique
|
|
$ExPrivModifiedLastnShareCount = $ExPrivModifiedLastnShare | Measure | select count -ExpandProperty count
|
|
|
|
# Percent of shares Modifieded in last n days
|
|
$ExpPrivModifiedLast = $ExPrivModifiedLastnShareCount / $AllSMBSharesCount
|
|
$ExpPrivModifiedLastP = $ExpPrivModifiedLast.tostring("P") -replace(" ","")
|
|
|
|
# Get summary bar code - Need to extend for counts,%, and bar
|
|
$ExPrivModifiedLastBars = Get-ExPrivSumData -DataTable $ExPrivModifiedLastn -AllComputerCount $ComputerCount -AllShareCount $AllSMBSharesCount -AllAclCount $ShareACLsCount
|
|
$ExPrivModifiedLastComputerB = $ExPrivModifiedLastBars.ComputerBar
|
|
$ExPrivModifiedLastShareB = $ExPrivModifiedLastBars.ShareBar
|
|
$ExPrivModifiedLastShareAclB = $ExPrivModifiedLastBars.AclBar
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Identified shares modified in last $LastModDays days."
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Identify affected subnets
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Get list of Subnets
|
|
$Subnets = $ExcessiveSharePrivs | Select IPAddress -Unique |
|
|
Foreach{
|
|
|
|
[int]$LastOctStart = (($_.IPAddress | Select-String '\.' -AllMatches).Matches | select -last 1 | select index -ExpandProperty index)
|
|
$Subnet = $_.IPAddress.substring(0,$LastOctStart)
|
|
$Subnet
|
|
} | select -Unique
|
|
|
|
$SubnetsCount = $Subnets | measure | select count -ExpandProperty count
|
|
|
|
# Get information for each subnet
|
|
$SubnetSummary = $Subnets |
|
|
foreach {
|
|
|
|
$Subnet = $_
|
|
$Subnetdisplay = $subnet + ".0"
|
|
|
|
# Acls - acl list exists
|
|
$subnetacls = $ExcessiveSharePrivs | where ipaddress -like "$subnet*"
|
|
$subnetaclsCount = $subnetacls | measure | select count -ExpandProperty count
|
|
|
|
# ACLs: Read - aclread exists
|
|
$subnetaclr = $SharesWithread | where ipaddress -like "$subnet*"
|
|
$subnetaclrCount = $subnetaclr | measure | select count -ExpandProperty count
|
|
|
|
# ACLs: Write - acl write exists
|
|
$subnetaclw = $SharesWithWrite | where ipaddress -like "$subnet*"
|
|
$subnetaclwCount = $subnetaclw | measure | select count -ExpandProperty count
|
|
|
|
# ACLs: Highrisk - acl highrisk exists
|
|
$subnetaclx = $SharesHighRisk | where ipaddress -like "$subnet*"
|
|
$subnetaclxCount = $subnetaclx | measure | select count -ExpandProperty count
|
|
|
|
# Shares
|
|
$subnetshares = $subnetacls | select sharepath -Unique
|
|
$subnetsharesCount = $subnetshares | measure | select count -ExpandProperty count
|
|
|
|
# Computers
|
|
$subnetcomputers = $subnetacls | select computername -Unique
|
|
$subnetcomputersCount = $subnetcomputers | measure | select count -ExpandProperty count
|
|
|
|
# Check for known subnet information
|
|
if(-not $HostList){
|
|
if($DomainSubnets -ne 0){
|
|
|
|
$DomainSubnets |
|
|
foreach{
|
|
|
|
$SubnetFull = $_.Subnet
|
|
|
|
if((checkSubnet $SubnetFull $Subnet).condition){
|
|
$SubnetDesc = $_.Description
|
|
$SubnetCreated = $_.whencreated
|
|
$SubnetSite = $_.Site
|
|
}
|
|
}
|
|
}else{
|
|
$SubnetDesc = "Unknown"
|
|
$SubnetCreated = "Unknown"
|
|
$SubnetSite = "Unknown"
|
|
}
|
|
}else{
|
|
$SubnetDesc = "Unknown"
|
|
$SubnetCreated = "Unknown"
|
|
$SubnetSite = "Unknown"
|
|
}
|
|
|
|
# Create object
|
|
$Object = new-object PSObject
|
|
$Object | Add-Member Subnet $Subnetdisplay
|
|
$Object | Add-Member Desc $SubnetDesc
|
|
$Object | Add-Member Created $SubnetCreated
|
|
$Object | Add-Member Site $SubnetSite
|
|
$Object | Add-Member Acls $subnetaclsCount
|
|
$Object | Add-Member ReadAcls $subnetaclrCount
|
|
$Object | Add-Member WriteAcls $subnetaclwCount
|
|
$Object | Add-Member HighRiskAcls $subnetaclxCount
|
|
$Object | Add-Member Shares $subnetsharesCount
|
|
$Object | Add-Member Computers $subnetcomputersCount
|
|
|
|
# Return object
|
|
if($Subnetdisplay -ne ".0"){
|
|
$Object
|
|
}
|
|
|
|
} | sort Acls -Descending
|
|
|
|
# Status User
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] - Identified $SubnetsCount subnets hosting shares configured with excessive privileges."
|
|
$SubnetSummary | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Shares-Inventory-Common-Subnets.csv"
|
|
$SubnetFile = "$TargetDomain-Shares-Inventory-Common-Subnets.csv"
|
|
|
|
# Create HTML table for report
|
|
|
|
# Setup HTML begin
|
|
Write-Verbose "[+] Creating html top."
|
|
$HTMLSTART = @"
|
|
<table class="table table-striped table-hover tabledrop">
|
|
"@
|
|
|
|
# Get list of columns
|
|
$MyCsvColumns = ("Computers","Shares","HighRiskAcls","WriteAcls","ReadAcls","Acls","Site","Created","Desc","Subnet")
|
|
|
|
# Print columns creation
|
|
$HTMLTableHeadStart= "<thead><tr>"
|
|
$MyCsvColumns |
|
|
ForEach-Object {
|
|
|
|
# Add column
|
|
$HTMLTableColumn = "<th>$_</th>$HTMLTableColumn"
|
|
}
|
|
$HTMLTableColumn = "$HTMLTableHeadStart$HTMLTableColumn</tr></thead>"
|
|
|
|
# Create table rows
|
|
Write-Verbose "[+] Creating html table rows."
|
|
$HTMLTableRow = $SubnetSummary |
|
|
ForEach-Object {
|
|
|
|
# Create a value contain row data
|
|
$CurrentRow = $_
|
|
$PrintRow = ""
|
|
$MyCsvColumns |
|
|
ForEach-Object{
|
|
|
|
try{
|
|
$GetValue = $CurrentRow | Select-Object $_ -ExpandProperty $_ -ErrorAction SilentlyContinue
|
|
if($PrintRow -eq ""){
|
|
$PrintRow = "<td>$GetValue</td>"
|
|
}else{
|
|
$PrintRow = "<td>$GetValue</td>$PrintRow"
|
|
}
|
|
}catch{}
|
|
}
|
|
|
|
# Return row
|
|
$HTMLTableHeadstart = "<tr>"
|
|
$HTMLTableHeadend = "</tr>"
|
|
"$HTMLTableHeadStart$PrintRow$HTMLTableHeadend"
|
|
}
|
|
|
|
# Setup HTML end
|
|
Write-Verbose "[+] Creating html bottom."
|
|
$HTMLEND = @"
|
|
</tbody>
|
|
</table>
|
|
"@
|
|
|
|
# Return it
|
|
$SubnetSummaryHTML = "$HTMLSTART $HTMLTableColumn $HTMLTableRow $HTMLEND"
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Calculate percentages
|
|
# ----------------------------------------------------------------------
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
#Write-Output " [*][$Time] Generating Report:"
|
|
|
|
# Set good/bad images
|
|
$CheckBad = '<img style="padding-top:5px;padding-left:20px;" src="data:image/png;base64, iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAAACXBIWXMAAAGPAAABjwEeLVWuAAAAGXRFWHRTb2Z0d2FyZQB3d3cuaW5rc2NhcGUub3Jnm+48GgAAAkVJREFUOI2FlLtLW3EUx78akiiCLuIgKBh1NzjEIQ4SRRShxRofiLOga/8FoakVxEUnFx91Egpurg6Kf0DS1lbULb6CV40PyKfD797eXO9Ne+As53fO597zlAIEKYI0gbSNlEO6tzVn28aRIkGxQbAPSL+QIBSCeBxGRox2dRmbBNIJ0ui/QCGkL0jQ1ATLy3B5iU/yeVhagsZGB7yIVB0ENLDBQbi99YPeytUVpFIONBOUJgwNwevr/2GOvLxAf78DfV/egBOam6FQcJ1PT2F11QsolSCT8WZwc2NKJP1GisjuJqytuU5nZ9DWZr68sODC5uaMLR6H62vXf2XF+cu0kL4SjcLdnXl8foaODsfBaCYDs7NeWyrlAgsFiERA2hTSD3p7vant7UE06gWUa0MDHB56Y3p6QMoJyWJ62l/wStAgGMD4OEiWmZ+qKv9gDg9LAwN++9SUlEj47aGQJJWE9J2+Pn835+crp+w0qlySSZCyQtqmthYeHirD6uogHK4MtSyoqQFpQ/aiw8aGeczn3ZFxanZ0BLu7XmgiYSYCYH3dsY8JKYz0k1gMikXjcH4O7e3+BjiN6u525/D+HlpbsQ9KxNmWUSRIp03KYIb7+Nhfq/19d6NKJZiYcP7u3dt9XkSCyUl4fPSD3kqxCDMzDuxT0LWpRvqMBJ2dsLPj1qhcnp5gawtisb+Xpvx8+QbQvhqLkjpUXy8lk1JLiwm9uJAODiTLkqQTSR+rpG/+ofRDw0hppE2kLJJla9a2jSGFg2L/AOl+fNdFEbGxAAAAAElFTkSuQmCC" />'
|
|
$CheckGood = '<img style="padding-top:5px;padding-left:20px;" src="data:image/png;base64, iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAAACXBIWXMAAAGPAAABjwEeLVWuAAAAGXRFWHRTb2Z0d2FyZQB3d3cuaW5rc2NhcGUub3Jnm+48GgAAAudJREFUOI2Nld1LU3EYxz/nnLVJpCMKQ6eTNESCDCy6MXoznEKglZIYddFNF91GFxF1KojyLYVuutKcFMyQ7Q8IFkWtsBDfIip68QWbwnzbi1vb08Xcam6W37tznvP9nOf3+/F9fgqZpGNE4SRCHVABFKxWJoH3gBNwohNea1XSYDc5jdACFGuKRvmOciw5ljhtcZKRnyNEJQrwBbiCzkDGpnCgodOGjuS25kqnp1Nm/bOyVl6/V9pftcv2lu2CjqDTio6aaZlt6IjNbhNf0JcGWqu5wJxUPapKQO+lL1NHavtqJRKN/BeWUDgaluO9xxPQ+vge6hiB8fzs/JLxS+OYTebMW7KOfCEfZQ/K8Pq9X4EyFYWTQMn1w9c3BFtYWeCC6wJzgTkAtmZt5dqhawA7UahTEepNmonmPc0bgtnsNrqHuqnqrUpCz+89j1EzglCnAvsOWA6Qbcz+J2w+NE+1vZo3U28AGP45TMfrDgDMJjMVeRUA+1Ugz2q2ppj9ET+2Phuuj65kZzV9Nbydepv8pmF3A7eO3ko+rzIsBgBFUVJgJx6fwP3Njfubm576Hro8XcnOErAnp59gUA3Jd5qiAcQMwPTU4lRpojA0M4Rn0gNAOBrm7MBZROSfMICJxQmAaRV455n0EIgEAKgsrMTV5CLLkAWwIdhyeJnB6UGAQRVwBn8FGfjwJ5LVJdXcPnY7xVRkLsoIA+gf7yf0KwTggodsQudTcVexBCPBlCTceXFH0JGCjoK0WkLL4WWx3rcKOl9WQwLonEJHGh2NEovFUgz2Yfu6sFgsJmf6z8Sjd5O61L7jU0OanjZJIBL4b46DkaCcGziXyPHd5GkngUd4Bmwe9Y5WOsYc5G7JpXRbKZqqpfx3JbqCY8xBQ38Dz78/B2gBruJGINOAjU+NVmBXjimHg9aDFOYUIggTCxO8/PGSpfASwGcULnMD19/2dCDED2qG+tUrYB+pV8A7wEkeLi4SWWv9DTik/yQF2VYdAAAAAElFTkSuQmCC" />'
|
|
|
|
# top 5 shares
|
|
$DupDec = $Top5ShareCountTotal / $AllAccessibleSharesCount
|
|
$DupPercent = $DupDec.tostring("P")
|
|
|
|
# Expected share count from know defaults
|
|
$MinExpectedShareCount = $Computers445OpenCount * 2
|
|
|
|
# Computer ping
|
|
$PercentComputerPing = [math]::Round($ComputerPingableCount/$ComputerCount,4)
|
|
$PercentComputerPingP = $PercentComputerPing.tostring("P") -replace(" ","")
|
|
$PercentComputerPingBarVal = ($PercentComputerPing*2).tostring("P") -replace(" %","px")
|
|
|
|
# Computer port 445 open
|
|
$PercentComputerPort = [math]::Round($Computers445OpenCount/$ComputerCount,4)
|
|
$PercentComputerPortP = $PercentComputerPort.tostring("P") -replace(" ","")
|
|
$PercentComputerPortBarVal = ($PercentComputerPort*2).tostring("P") -replace(" %","")
|
|
|
|
# Computer with share
|
|
$PercentComputerWitShare = [math]::Round($AllComputersWithSharesCount/$ComputerCount,4)
|
|
$PercentComputerWitShareP = $PercentComputerWitShare.tostring("P") -replace(" ","")
|
|
$PercentComputerWitShareBarVal = ($PercentComputerWitShare*2).tostring("P") -replace(" %","px")
|
|
|
|
# Computer with non default shares
|
|
$PercentComputerNonDefault = [math]::Round($ComputerwithNonDefaultCount/$ComputerCount,4)
|
|
$PercentComputerNonDefaultP = $PercentComputerNonDefault.tostring("P") -replace(" ","")
|
|
$PercentComputerNonDefaultBarVal = ($PercentComputerNonDefault*2).tostring("P") -replace(" %","px")
|
|
|
|
# Computer with excessive priv shares
|
|
$PercentComputerExPriv = [math]::Round($ComputerWithExcessive/$ComputerCount,4)
|
|
$PercentComputerExPrivP = $PercentComputerExPriv.tostring("P") -replace(" ","")
|
|
$PercentComputerExPrivBarVal = ($PercentComputerExPriv*2).tostring("P") -replace(" %","px")
|
|
|
|
# Computer read share access
|
|
$PercentComputerRead = [math]::Round($ComputerWithReadCount/$ComputerCount,4)
|
|
$PercentComputerReadP = $PercentComputerRead.tostring("P") -replace(" ","")
|
|
$PercentComputerReadBarVal = ($PercentComputerRead*2).tostring("P") -replace(" %","px")
|
|
if($PercentComputerRead -ne 0){
|
|
$CheckStatusComputerR = $CheckBad
|
|
}else{
|
|
$CheckStatusComputerR = $CheckGood
|
|
}
|
|
|
|
# Computer write share access
|
|
$PercentComputerWrite = [math]::Round($ComputerWithWriteCount/$ComputerCount,4)
|
|
$PercentComputerWriteP = $PercentComputerWrite.tostring("P") -replace(" ","")
|
|
$PercentComputerWriteBarVal = ($PercentComputerWrite*2).tostring("P") -replace(" %","px")
|
|
if($PercentComputerWrite -ne 0){
|
|
$CheckStatusComputerW = $CheckBad
|
|
}else{
|
|
$CheckStatusComputerW = $CheckGood
|
|
}
|
|
|
|
# Computer highrisk shares
|
|
$PercentComputerHighRisk = [math]::Round($ComputerwithHighRisk/$ComputerCount,4)
|
|
$PercentComputerHighRiskP = $PercentComputerHighRisk.tostring("P") -replace(" ","")
|
|
$PercentComputerHighRiskBarVal = ($PercentComputerHighRisk*2).tostring("P") -replace(" %","px")
|
|
if($PercentComputerHighRisk -ne 0){
|
|
$CheckStatusComputerH = $CheckBad
|
|
}else{
|
|
$CheckStatusComputerH = $CheckGood
|
|
}
|
|
|
|
# Shares with non default names
|
|
$PercentSharesNonDefault = [math]::Round($SharesNonDefaultCount/$AllSMBSharesCount,4)
|
|
$PercentSharesNonDefaultP = $PercentSharesNonDefault.tostring("P") -replace(" ","")
|
|
$PercentSharesNonDefaultBarVal = ($PercentSharesNonDefault*2).tostring("P") -replace(" %","px")
|
|
|
|
# Shares with excessive priv shares
|
|
$PercentSharesExPriv = [math]::Round($ExcessiveSharesCount/$AllSMBSharesCount,4)
|
|
$PercentSharesExPrivP = $PercentSharesExPriv.tostring("P") -replace(" ","")
|
|
$PercentSharesExPrivBarVal = ($PercentSharesExPriv*2).tostring("P") -replace(" %","px")
|
|
|
|
# Shares with excessive read
|
|
$PercentSharesRead = [math]::Round($SharesWithReadCount/$AllSMBSharesCount,4)
|
|
$PercentSharesReadP = $PercentSharesRead.tostring("P") -replace(" ","")
|
|
$PercentSharesReadBarVal = ($PercentSharesRead*2).tostring("P") -replace(" %","px")
|
|
if($PercentSharesRead -ne 0){
|
|
$CheckStatusShareR = $CheckBad
|
|
}else{
|
|
$CheckStatusShareR = $CheckGood
|
|
}
|
|
|
|
# Shares with excessive write
|
|
$PercentSharesWrite = [math]::Round($SharesWithWriteCount/$AllSMBSharesCount,4)
|
|
$PercentSharesWriteP = $PercentSharesWrite.tostring("P") -replace(" ","")
|
|
$PercentSharesWriteBarVal = ($PercentSharesWrite*2).tostring("P") -replace(" %","px")
|
|
if($PercentSharesWrite -ne 0){
|
|
$CheckStatusShareW = $CheckBad
|
|
}else{
|
|
$CheckStatusShareW = $CheckGood
|
|
}
|
|
|
|
# Shares with excessive highrisk
|
|
$PercentSharesHighRisk = [math]::Round($SharesHighRiskCount/$AllSMBSharesCount,4)
|
|
$PercentSharesHighRiskP = $PercentSharesHighRisk.tostring("P") -replace(" ","")
|
|
$PercentSharesHighRiskBarVal = ($PercentSharesHighRisk*2).tostring("P") -replace(" %","px")
|
|
if($PercentSharesHighRisk -ne 0){
|
|
$CheckStatusShareH = $CheckBad
|
|
}else{
|
|
$CheckStatusShareH = $CheckGood
|
|
}
|
|
|
|
# ACL with non default names
|
|
$PercentAclNonDefault = [math]::Round($AclNonDefaultCount/$ShareACLsCount,4)
|
|
$PercentAclNonDefaultP = $PercentAclNonDefault.tostring("P") -replace(" ","")
|
|
$PercentAclNonDefaultBarVal = ($PercentAclNonDefault*2).tostring("P") -replace(" %","px")
|
|
|
|
# ACL with excessive priv shares
|
|
$PercentAclExPriv = [math]::Round($ExcessiveSharePrivsCount/$ShareACLsCount,4)
|
|
$PercentAclExPrivP = $PercentAclExPriv.tostring("P") -replace(" ","")
|
|
$PercentAclExPrivBarVal = ($PercentAclExPriv*2).tostring("P") -replace(" %","px")
|
|
|
|
# ACL with excessive read
|
|
$PercentAclRead = [math]::Round($AclWithReadCount/$ShareACLsCount,4)
|
|
$PercentAclReadP = $PercentAclRead.tostring("P") -replace(" ","")
|
|
$PercentAclReadBarVal = ($PercentAclRead *2).tostring("P") -replace(" %","px")
|
|
if($PercentAclRead -ne 0){
|
|
$CheckStatusAclR = $CheckBad
|
|
}else{
|
|
$CheckStatusAclR = $CheckGood
|
|
}
|
|
|
|
# ACL with excessive write
|
|
$PercentAclWrite = [math]::Round($AclWithWriteCount/$ShareACLsCount,4)
|
|
$PercentAclWriteP = $PercentAclWrite.tostring("P") -replace(" ","")
|
|
$PercentAclWriteBarVal = ($PercentAclWrite *2).tostring("P") -replace(" %","px")
|
|
if($PercentAclWrite -ne 0){
|
|
$CheckStatusAclW = $CheckBad
|
|
}else{
|
|
$CheckStatusAclW = $CheckGood
|
|
}
|
|
|
|
# ACL with excessive highrisk
|
|
$PercentAclHighRisk = [math]::Round($AclHighRiskCount/$ShareACLsCount,4)
|
|
$PercentAclHighRiskP = $PercentAclHighRisk.tostring("P") -replace(" ","")
|
|
$PercentAclHighRiskBarVal = ($PercentAclHighRisk *2).tostring("P") -replace(" %","px")
|
|
if($PercentAclHighRisk -ne 0){
|
|
$CheckStatusAclH = $CheckBad
|
|
}else{
|
|
$CheckStatusAclH = $CheckGood
|
|
}
|
|
|
|
# ACE User: Everyone
|
|
$AceEveryone = Get-UserAceCounts -DataTable $ExcessiveSharePrivs -UserName "everyone"
|
|
$AceEveryoneAclCount = $AceEveryone.UserAclsCount
|
|
$AceEveryoneShareCount = $AceEveryone.UserShareCount
|
|
$AceEveryoneComputerCount = $AceEveryone.UserComputerCount
|
|
$AceEveryoneAclReadCount = $AceEveryone.UserReadAclCount
|
|
$AceEveryoneAclWriteCount = $AceEveryone.UserWriteAclCount
|
|
$AceEveryoneAclHRCount = $AceEveryone.UserHighRiskAclCount
|
|
|
|
$AceEveryoneAclP = Get-PercentDisplay -TargetCount $AceEveryoneComputerCount -FullCount $ComputerCount
|
|
$AceEveryoneAclPS = $AceEveryoneAclP.PercentString
|
|
$AceEveryoneAclPB = $AceEveryoneAclP.PercentBarVal
|
|
|
|
$AceEveryoneShareCountP = Get-PercentDisplay -TargetCount $AceEveryoneShareCount -FullCount $AllSMBSharesCount
|
|
$AceEveryoneShareCountPS = $AceEveryoneShareCountP.PercentString
|
|
$AceEveryoneShareCountPB = $AceEveryoneShareCountP.PercentBarVal
|
|
|
|
$AceEveryoneComputerCountP = Get-PercentDisplay -TargetCount $AceEveryoneAclCount -FullCount $ShareACLsCount
|
|
$AceEveryoneComputerCountPS = $AceEveryoneComputerCountP.PercentString
|
|
$AceEveryoneComputerCountPB = $AceEveryoneComputerCountP.PercentBarVal
|
|
|
|
# ACE User: Users
|
|
$AceUsers = Get-UserAceCounts -DataTable $ExcessiveSharePrivs -UserName "BUILTIN\Users"
|
|
$AceUsersAclCount = $AceUsers.UserAclsCount
|
|
$AceUsersShareCount = $AceUsers.UserShareCount
|
|
$AceUsersComputerCount = $AceUsers.UserComputerCount
|
|
$AceUsersAclReadCount = $AceUsers.UserReadAclCount
|
|
$AceUsersAclWriteCount = $AceUsers.UserWriteACLCount
|
|
$AceUsersAclHRCount = $AceUsers.UserHighRiskACLCount
|
|
|
|
$AceUsersAclP = Get-PercentDisplay -TargetCount $AceUsersComputerCount -FullCount $ComputerCount
|
|
$AceUsersAclPS = $AceUsersAclP.PercentString
|
|
$AceUsersAclPB = $AceUsersAclP.PercentBarVal
|
|
|
|
$AceUsersShareCountP = Get-PercentDisplay -TargetCount $AceUsersShareCount -FullCount $AllSMBSharesCount
|
|
$AceUsersShareCountPS = $AceUsersShareCountP.PercentString
|
|
$AceUsersShareCountPB = $AceUsersShareCountP.PercentBarVal
|
|
|
|
$AceUsersComputerCountP = Get-PercentDisplay -TargetCount $AceUsersAclCount -FullCount $ShareACLsCount
|
|
$AceUsersComputerCountPS = $AceUsersComputerCountP.PercentString
|
|
$AceUsersComputerCountPB = $AceUsersComputerCountP.PercentBarVal
|
|
|
|
# ACE User: Authenticated Users
|
|
$AceAuthenticatedUsers = Get-UserAceCounts -DataTable $ExcessiveSharePrivs -UserName "Authenticated Users"
|
|
$AceAuthenticatedUsersComputerCount = $AceAuthenticatedUsers.UserComputerCount
|
|
$AceAuthenticatedUsersShareCount = $AceAuthenticatedUsers.UserShareCount
|
|
$AceAuthenticatedUsersAclCount = $AceAuthenticatedUsers.UserAclsCount
|
|
$AceAuthenticatedUsersAclReadCount = $AceAuthenticatedUsers.UserReadAclCount
|
|
$AceAuthenticatedUsersAclWriteCount = $AceAuthenticatedUsers.UserWriteACLCount
|
|
$AceAuthenticatedUsersAclHRCount = $AceAuthenticatedUsers.UserHighRiskACLCount
|
|
|
|
$AceAuthenticatedUsersAclP = Get-PercentDisplay -TargetCount $AceAuthenticatedUsersComputerCount -FullCount $ComputerCount
|
|
$AceAuthenticatedUsersAclPS = $AceAuthenticatedUsersAclP.PercentString
|
|
$AceAuthenticatedUsersAclPB = $AceAuthenticatedUsersAclP.PercentBarVal
|
|
|
|
$AceAuthenticatedUsersShareCountP = Get-PercentDisplay -TargetCount $AceAuthenticatedUsersShareCount -FullCount $AllSMBSharesCount
|
|
$AceAuthenticatedUsersShareCountPS = $AceAuthenticatedUsersShareCountP.PercentString
|
|
$AceAuthenticatedUsersShareCountPB = $AceAuthenticatedUsersShareCountP.PercentBarVal
|
|
|
|
$AceAuthenticatedUsersComputerCountP = Get-PercentDisplay -TargetCount $AceAuthenticatedUsersAclCount -FullCount $ShareACLsCount
|
|
$AceAuthenticatedUsersComputerCountPS = $AceAuthenticatedUsersComputerCountP.PercentString
|
|
$AceAuthenticatedUsersComputerCountPB = $AceAuthenticatedUsersComputerCountP.PercentBarVal
|
|
|
|
# ACE User: Domain Users
|
|
$AceDomainUsers = Get-UserAceCounts -DataTable $ExcessiveSharePrivs -UserName "Domain Users"
|
|
$AceDomainUsersAclCount = $AceDomainUsers.UserAclsCount
|
|
$AceDomainUsersShareCount = $AceDomainUsers.UserShareCount
|
|
$AceDomainUsersComputerCount = $AceDomainUsers.UserComputerCount
|
|
$AceDomainUsersAclReadCount = $AceDomainUsers.UserReadAclCount
|
|
$AceDomainUsersAclWriteCount = $AceDomainUsers.UserWriteACLCount
|
|
$AceDomainUsersAclHRCount = $AceDomainUsers.UserHighRiskACLCount
|
|
|
|
$AceDomainUsersAclP = Get-PercentDisplay -TargetCount $AceDomainUsersComputerCount -FullCount $ComputerCount
|
|
$AceDomainUsersAclPS = $AceDomainUsersAclP.PercentString
|
|
$AceDomainUsersAclPB = $AceDomainUsersAclP.PercentBarVal
|
|
|
|
$AceDomainUsersShareCountP = Get-PercentDisplay -TargetCount $AceDomainUsersShareCount -FullCount $AllSMBSharesCount
|
|
$AceDomainUsersShareCountPS = $AceDomainUsersShareCountP.PercentString
|
|
$AceDomainUsersShareCountPB = $AceDomainUsersShareCountP.PercentBarVal
|
|
|
|
$AceDomainUsersComputerCountP = Get-PercentDisplay -TargetCount $AceDomainUsersAclCount -FullCount $ShareACLsCount
|
|
$AceDomainUsersComputerCountPS = $AceDomainUsersComputerCountP.PercentString
|
|
$AceDomainUsersComputerCountPB = $AceDomainUsersComputerCountP.PercentBarVal
|
|
|
|
# ACE User: Domain Computers
|
|
$AceDomainComputers = Get-UserAceCounts -DataTable $ExcessiveSharePrivs -UserName "Domain Computers"
|
|
$AceDomainComputersAclCount = $AceDomainComputers.UserAclsCount
|
|
$AceDomainComputersShareCount = $AceDomainComputers.UserShareCount
|
|
$AceDomainComputersComputerCount = $AceDomainComputers.UserComputerCount
|
|
$AceDomainComputersAclReadCount = $AceDomainComputers.UserReadAclCount
|
|
$AceDomainComputersAclWriteCount = $AceDomainComputers.UserWriteACLCount
|
|
$AceDomainComputersAclHRCount = $AceDomainComputers.UserHighRiskACLCount
|
|
|
|
$AceDomainComputersAclP = Get-PercentDisplay -TargetCount $AceDomainComputersComputerCount -FullCount $ComputerCount
|
|
$AceDomainComputersAclPS = $AceDomainComputersAclP.PercentString
|
|
$AceDomainComputersAclPB = $AceDomainComputersAclP.PercentBarVal
|
|
|
|
$AceDomainComputersShareCountP = Get-PercentDisplay -TargetCount $AceDomainComputersShareCount -FullCount $AllSMBSharesCount
|
|
$AceDomainComputersShareCountPS = $AceDomainComputersShareCountP.PercentString
|
|
$AceDomainComputersShareCountPB = $AceDomainComputersShareCountP.PercentBarVal
|
|
|
|
$AceDomainComputersComputerCountP = Get-PercentDisplay -TargetCount $AceDomainComputersAclCount -FullCount $ShareACLsCount
|
|
$AceDomainComputersComputerCountPS = $AceDomainComputersComputerCountP.PercentString
|
|
$AceDomainComputersComputerCountPB = $AceDomainComputersComputerCountP.PercentBarVal
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
#Write-Output " [*][$Time] - Summary report data generated."
|
|
#Write-Output " [*][$Time] - $Top5ShareCountTotal of $AllAccessibleSharesCount ($DupPercent) shares are associated with the top 5 share names."
|
|
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Create Timeline Reports
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Generate last modified card
|
|
If($SupressTimelineRpt){
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Creation timeline reports have been disabled."
|
|
$CardLastModifiedTimeLine = "Share creation Timeline reports have been disabled."
|
|
}else{
|
|
$CardCreationTimeLine = Get-CardCreationTime -MyDataTable $ExcessiveSharePrivs -OutFilePath "$OutputDirectory\$TargetDomain-Shares-Timeline-Creation-Summary.csv"
|
|
}
|
|
|
|
# Generate last modified card
|
|
If($SupressTimelineRpt){
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Last modified timeline reports have been disabled."
|
|
$CardLastModifiedTimeLine = "Last modified timeline reports have been disabled."
|
|
}else{
|
|
$CardLastModifiedTimeLine = Get-CardLastModified -MyDataTable $ExcessiveSharePrivs -OutFilePath "$OutputDirectory\$TargetDomain-Shares-Timeline-Last-Modified-Summary.csv"
|
|
}
|
|
|
|
# Generate last access card
|
|
If($SupressTimelineRpt){
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Output " [*][$Time] Last access timeline reports have been disabled."
|
|
$CardLastAccessTimeLine = "Last access timeline reports have been disabled."
|
|
}else{
|
|
$CardLastAccessTimeLine = Get-CardLastAccess -MyDataTable $ExcessiveSharePrivs -OutFilePath "$OutputDirectory\$TargetDomain-Shares-Timeline-Last-Accessed-Summary.csv"
|
|
}
|
|
|
|
Write-Output " [*][$Time] Analysis Complete"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Display final summary
|
|
# ----------------------------------------------------------------------
|
|
|
|
Write-Output " ---------------------------------------------------------------"
|
|
Write-Output " SHARE REPORT SUMMARY "
|
|
Write-Output " ---------------------------------------------------------------"
|
|
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
#Write-Output " [*][$Time] Results written to $OutputDirectory"
|
|
#Write-Output " [*][$Time] "
|
|
|
|
$EndTime = Get-Date
|
|
$StopWatch.Stop()
|
|
$RunTime = $StopWatch | Select-Object Elapsed -ExpandProperty Elapsed
|
|
|
|
if($NoPing){
|
|
$NoPingMsg = "(No Ping)"
|
|
}else{
|
|
$NoPingMsg = ""
|
|
}
|
|
|
|
#Write-Output " [*][$Time] -----------------------------------------------"
|
|
#Write-Output " [*][$Time] Get-ShareInventory Summary Report"
|
|
#Write-Output " [*][$Time] -----------------------------------------------"
|
|
Write-Output " [*][$Time] Domain: $TargetDomain"
|
|
Write-Output " [*][$Time] Start time: $StartTime"
|
|
Write-Output " [*][$Time] End time: $EndTime"
|
|
Write-Output " [*][$Time] Run time: $RunTime"
|
|
Write-Output " [*][$Time] "
|
|
Write-Output " [*][$Time] COMPUTER SUMMARY"
|
|
Write-Output " [*][$Time] - $ComputerCount domain computers found."
|
|
Write-Output " [*][$Time] - $ComputerPingableCount ($PercentComputerPingP) domain computers responded to ping. $NoPingMsg"
|
|
Write-Output " [*][$Time] - $Computers445OpenCount ($PercentComputerPortP) domain computers had TCP port 445 accessible."
|
|
Write-Output " [*][$Time] - $ComputerwithNonDefaultCount ($PercentComputerNonDefaultP) domain computers had shares that were non-default."
|
|
Write-Output " [*][$Time] - $ComputerWithExcessive ($PercentComputerExPrivP) domain computers had shares with potentially excessive privileges."
|
|
Write-Output " [*][$Time] - $ComputerWithReadCount ($PercentComputerReadP) domain computers had shares that allowed READ access."
|
|
Write-Output " [*][$Time] - $ComputerWithWriteCount ($PercentComputerWriteP) domain computers had shares that allowed WRITE access."
|
|
Write-Output " [*][$Time] - $ComputerwithHighRisk ($PercentComputerHighRiskP) domain computers had shares that are HIGH RISK."
|
|
Write-Output " [*][$Time] "
|
|
Write-Output " [*][$Time] SHARE SUMMARY"
|
|
Write-Output " [*][$Time] - $AllSMBSharesCount shares were found. We expect a minimum of $MinExpectedShareCount shares"
|
|
Write-Output " [*][$Time] because $Computers445OpenCount systems had open ports and there are typically two default shares."
|
|
Write-Output " [*][$Time] - $SharesNonDefaultCount ($PercentSharesNonDefaultP) shares across $ComputerwithNonDefaultCount systems were non-default."
|
|
Write-Output " [*][$Time] - $ExcessiveSharesCount ($PercentSharesExPrivP) shares across $ComputerWithExcessive systems are configured with $ExcessiveSharePrivsCount potentially excessive ACLs."
|
|
Write-Output " [*][$Time] - $SharesWithReadCount ($PercentSharesReadP) shares across $ComputerWithReadCount systems allowed READ access."
|
|
Write-Output " [*][$Time] - $SharesWithWriteCount ($PercentSharesWriteP) shares across $ComputerWithWriteCount systems allowed WRITE access."
|
|
Write-Output " [*][$Time] - $SharesHighRiskCount ($PercentSharesHighRiskP) shares across $ComputerwithHighRisk systems are considered HIGH RISK."
|
|
Write-Output " [*][$Time] "
|
|
Write-Output " [*][$Time] SHARE ACL SUMMARY"
|
|
Write-Output " [*][$Time] - $ShareACLsCount ACLs were found."
|
|
Write-Output " [*][$Time] - $AclNonDefaultCount ($PercentAclNonDefaultP) ACLs were associated with non-default shares."
|
|
Write-Output " [*][$Time] - $ExcessiveSharePrivsCount ($PercentAclExPrivP) ACLs were found to be potentially excessive."
|
|
Write-Output " [*][$Time] - $AclWithReadCount ($PercentAclReadP) ACLs were found that allowed READ access."
|
|
Write-Output " [*][$Time] - $AclWithWriteCount ($PercentAclWriteP) ACLs were found that allowed WRITE access."
|
|
Write-Output " [*][$Time] - $AclHighRiskCount ($PercentAclHighRiskP) ACLs were found that are associated with HIGH RISK share names."
|
|
Write-Output " [*][$Time] "
|
|
Write-Output " [*][$Time] - The 5 most common share names are:"
|
|
Write-Output " [*][$Time] - $Top5ShareCountTotal of $AllAccessibleSharesCount ($DupPercent) discovered shares are associated with the top 5 share names."
|
|
$CommonShareNamesTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$ShareName = $_.name
|
|
Write-Output " [*][$Time] - $ShareCount $ShareName"
|
|
}
|
|
Write-Output " [*] -----------------------------------------------"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Display final summary - NEW HTML REPORT
|
|
# ----------------------------------------------------------------------
|
|
if($username -like ""){$username = whoami}
|
|
$SourceIps = (Get-NetIPAddress | where AddressState -like "*Pref*" | where AddressFamily -like "ipv4" | where ipaddress -notlike "127.0.0.1" | select IpAddress).ipaddress -join ("<br>")
|
|
$SourceHost = (hostname)
|
|
|
|
# Get share list string list
|
|
$CommonShareFileGroupTopString = $CommonShareFileGroupTop5 |
|
|
foreach {
|
|
$FileGroupName = $_.name
|
|
$ThisFileBars = Get-GroupFileBar -DataTable $ExcessiveSharePrivs -Name $FileGroupName -AllComputerCount $ComputerCount -AllShareCount $AllSMBSharesCount -AllAclCount $ShareACLsCount
|
|
$ComputerBarF = $ThisFileBars.ComputerBar
|
|
$ShareBarF = $ThisFileBars.ShareBar
|
|
$AclBarF = $ThisFileBars.AclBar
|
|
$ThisFileListPrep = $ThisFileBars.FileList
|
|
$ThisFileList = $ThisFileListPrep -replace "`n", "<br>"
|
|
$ThisFileCount = $ThisFileBars.FileCount
|
|
$ThisFileShareCount = $ThisFileBars.Sharecount
|
|
$ThisFileShareNameList = $ExcessiveSharePrivs | where FileListGroup -eq $FileGroupName | select ShareName -unique -expandproperty sharename | foreach { "$_ <br>"}
|
|
$ThisRow = @"
|
|
<tr>
|
|
<td>
|
|
<button class="collapsible">$ThisFileShareCount</button>
|
|
<div class="content">
|
|
<div class="filelist" >
|
|
$ThisFileShareNameList
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
$FileGroupName
|
|
</td>
|
|
<td>
|
|
<button class="collapsible"><span style="color:#CE112D;"></span>$ThisFileCount Files</button>
|
|
<div class="content">
|
|
<div class="filelist" >
|
|
$ThisFileList
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
$ComputerBarF
|
|
</td>
|
|
<td>
|
|
$ShareBarF
|
|
</td>
|
|
<td>
|
|
$AclBarF
|
|
</td>
|
|
</tr>
|
|
"@
|
|
$ThisRow
|
|
}
|
|
|
|
# Get share name string list
|
|
$CommonShareNamesTopString = $CommonShareNamesTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$ShareName = $_.name
|
|
Write-Output "$ShareCount $ShareName <br>"
|
|
}
|
|
|
|
# Get share name string list table
|
|
$CommonShareNamesTopStringT = $CommonShareNamesTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$ShareName = $_.name
|
|
$ShareFolderGroupCount = $ExcessiveSharePrivs | where sharename -like "$ShareName" | select filelistgroup -Unique | measure | select count -ExpandProperty count
|
|
$ShareNameBars = Get-GroupNameNoBar -DataTable $ExcessiveSharePrivs -Name $ShareName -AllComputerCount $ComputerCount -AllShareCount $AllSMBSharesCount -AllAclCount $ShareACLsCount
|
|
$ComputerBar = $ShareNameBars.ComputerBar
|
|
$ShareBar = $ShareNameBars.ShareBar
|
|
$AclBar = $ShareNameBars.AclBar
|
|
|
|
# Share Description
|
|
$ShareDescriptionSample = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | where ShareDescription -NE "" | select ShareDescription -first 1 -expandproperty ShareDescription | foreach {"<strong>Sample Description</strong><br> $_ <br><br> "}
|
|
|
|
# First created
|
|
$ShareFirstCreated = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select creationdate | foreach{[datetime]$_.creationdate } | Sort-Object | select -First 1 | foreach {$_.tostring("MM/dd/yyyy HH:mm:ss")}
|
|
|
|
# Last created
|
|
$ShareLastCreated = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select creationdate | foreach{[datetime]$_.creationdate } | Sort-Object -Descending | select -First 1 | foreach {$_.tostring("MM/dd/yyyy HH:mm:ss")}
|
|
|
|
# Last modified
|
|
$ShareLastModified = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select LastModifiedDate | foreach{[datetime]$_.LastModifiedDate } | Sort-Object -Descending | select -First 1 | foreach {$_.tostring("MM/dd/yyyy HH:mm:ss")}
|
|
|
|
# Share owner list
|
|
$ShareOwnerList = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | Sort-Object | select ShareOwner -Unique -ExpandProperty ShareOwner
|
|
|
|
# Share owner list count
|
|
$ShareOwnerListCount = $ShareOwnerList | select -Unique | measure-object | select count -expandproperty count
|
|
|
|
# Share folder group list
|
|
$ShareFolderGroupList = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select ShareName,FileListGroup -Unique | Group-Object FileListGroup | sort count -Descending | select count, name |
|
|
foreach {
|
|
|
|
$fdcount = $_.count;
|
|
$fdname = $_.name;
|
|
|
|
If($fdname){
|
|
|
|
# Get file count
|
|
$FdFileCount = $ExcessiveSharePrivs | where FileListGroup -eq $fdname | select FileCount -First 1 -ExpandProperty FileCount -ErrorAction SilentlyContinue
|
|
|
|
# Get file list
|
|
$MyFdList = $ExcessiveSharePrivs | where FileListGroup -eq $fdname | select FileList -First 1 -ExpandProperty FileList -ErrorAction SilentlyContinue
|
|
$MyFdListBr = $MyFdList -replace "`n", "<br>"
|
|
|
|
$ThisFileDirList = @"
|
|
<strong style="font-size: 10px;">$fdcount</strong>
|
|
<button class="collapsible" style="font-size: 10px;">$fdname ($FdFileCount Files)</button>
|
|
<div class="content">
|
|
<div class="filelist" style="font-size: 10px;">
|
|
$MyFdListBr
|
|
</div>
|
|
</div>
|
|
"@
|
|
$ThisFileDirList
|
|
}
|
|
}
|
|
|
|
# ----
|
|
# Calculate similarity here - Start
|
|
# How do i deal with divid by 1...?
|
|
|
|
# Calculate share to file group ratio: 0 to 1
|
|
# min value = 0
|
|
# max value = number of shares
|
|
# 10 shares / 1 group = 10.0 - highest score when only one group - correct
|
|
# 10 shares / 5 groups = 2.0 - high score when fewer groups - correct
|
|
# 10 shares / 10 groups = 1.0 - lower score when more groups - correct
|
|
# 5 shares / 10 groups = 0.5 - lower score when more groups - correct
|
|
$SimularityCalcShareFg1 = [math]::Round($ShareCount/$ShareFolderGroupCount,4) # Original value
|
|
$SimularityCalcShareFg = $SimularityCalcShareFg1 / $ShareCount
|
|
|
|
# Calculate share to owner ration: 0 to 1 (to show difference between avg foldergroup ad overall)
|
|
$SimularityCalcShareOwner1 = [math]::Round($ShareCount/$ShareOwnerListCount,4)
|
|
$SimularityCalcShareOwner = $SimularityCalcShareOwner1 / $ShareCount
|
|
|
|
# Calculate file group to owner avg ratio avg per share name: 0 to 1
|
|
$FGtoOwnersRatios = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select ShareName,FileListGroup -Unique | Group-Object FileListGroup | sort count -Descending | select count, name |
|
|
foreach{
|
|
$FgName = $_.name
|
|
$FgCount = $_.count
|
|
|
|
# Get count of owners associated with file group
|
|
$FgNameOwnercount = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | where FileListGroup -EQ "$FgName" | select ShareOwner -Unique | measure | select count -expandproperty count
|
|
|
|
# Calculate file group to owners ratio
|
|
[math]::Round($FgCount/$FgNameOwnercount,4)
|
|
}
|
|
$FGtoOwnersRatiosSum = 0
|
|
$FGtoOwnersRatiosCount = $FGtoOwnersRatios | measure-object | select count -expandproperty count
|
|
$FGtoOwnersRatios |
|
|
Foreach{
|
|
$FGtoOwnersRatiosSum += $_
|
|
}
|
|
$SimularityCalcFGOwnerAvg = [math]::Round($FGtoOwnersRatiosSum/$FGtoOwnersRatiosCount,4)
|
|
|
|
# Caluatlate if at least 1 file group is 50% or greater: 0 or 1
|
|
# take total count ($ShareFolderGroupCount)
|
|
# divide the number of instances by individual
|
|
# foreach loop until yes.
|
|
$fiftyorgreater = 0
|
|
$ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select ShareName,FileListGroup -Unique | Group-Object FileListGroup | sort count -Descending | select count, name |
|
|
foreach{
|
|
|
|
# Get % the file group represents for the share
|
|
$fgpercentage = [math]::Round($_.count/$ShareFolderGroupCount,4)
|
|
|
|
# If it's 50% or great flip the bit
|
|
if($fgpercentage -ge .5){
|
|
$SimularityCalc50P = 1
|
|
}
|
|
}
|
|
|
|
# Calculate share to creation date ratio (just informational, not used in similarity score - for now)
|
|
$ShareCreateCount = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select creationdate -Unique | Measure-Object | Select-Object count -ExpandProperty count
|
|
$SimularityCalcCreateDate1 = [math]::Round($ShareCount/$ShareCreateCount,4)
|
|
$SimularityCalcCreateDate = $SimularityCalcCreateDate1 / $ShareCount
|
|
|
|
# Calculate share to modification date ratio (just informational, not used in similarity score - for now)
|
|
$ShareModifiedCount = $ExcessiveSharePrivs | where sharename -EQ "$ShareName" | select LastModifiedDate -Unique | Measure-Object | Select-Object count -ExpandProperty count
|
|
$SimularityCalcLastModDate1 = [math]::Round($ShareCount/$ShareModifiedCount,4)
|
|
$SimularityCalcLastModDate = $SimularityCalcLastModDate1 / $ShareCount
|
|
|
|
# Calculate combined similarity score
|
|
# WeightFileGroup = 4
|
|
# WeightFiftyP = 3
|
|
# WeightFgOwnerAvg = 2
|
|
# WeightCreate = 1
|
|
# WeightLastMod = 1
|
|
# condense into 0-1, low (0-.50), medium(.51-.80), high similary (.81-1)
|
|
|
|
$SimularityCalcShareFgFinal = $SimularityCalcShareFg * 4
|
|
$SimularityCalc50PFinal = $SimularityCalc50P * 3
|
|
$SimularityCalcFGOwnerAvgFinal = $SimularityCalcFGOwnerAvg * 2
|
|
$SimularityCalcCreateDateFinal = $SimularityCalcCreateDate * 1
|
|
$SimularityCalcLastModDateFinal = $SimularityCalcLastModDate * 1
|
|
|
|
# Max is 4 + 3 + 2 + 1 + 1 = 11; Min is 0
|
|
$SimilarityTotal = $SimularityCalcShareFgFinal + $SimularityCalc50PFinal + $SimularityCalcFGOwnerAvgFinal +$SimularityCalcCreateDateFinal + $SimularityCalcLastModDateFinal
|
|
$SimilarityScore = $SimilarityTotal / 11
|
|
$SimilarityScoreP1 = [math]::round(($SimilarityScore.tostring("P") -replace('%','')))
|
|
$SimilarityScoreP = "$SimilarityScoreP1%"
|
|
If($SimilarityScore -gt .80){ $SimLevel = "High"}
|
|
If($SimilarityScore -lt .80){ $SimLevel = "Medium"}
|
|
If($SimilarityScore -lt .50){ $SimLevel = "Low"}
|
|
|
|
|
|
# Calculate similarity here - End
|
|
# ----
|
|
|
|
$ThisRow = @"
|
|
<tr>
|
|
<td>
|
|
$ShareCount
|
|
</td>
|
|
<td style="vertical-align: top;">
|
|
<button class="collapsible">$ShareName</button>
|
|
<div class="content">
|
|
<div class="filelistparent" style="font-size: 10px;">
|
|
$ShareDescriptionSample
|
|
|
|
<strong>Timeline Context</strong><br>
|
|
<table class="subtable">
|
|
<tr>
|
|
<td>First Created:</td>
|
|
<td> $ShareFirstCreated</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Last Created:</td>
|
|
<td> $ShareLastCreated</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Last Modified:</td>
|
|
<td> $ShareLastModified</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<button class="collapsible" style="font-size: 10px;"><strong>$SimLevel ($SimilarityScoreP)</strong></button>
|
|
<div class="content">
|
|
<div class="filelist" style="font-size: 10px;">
|
|
<strong>Normalized Ratio Details</strong><br>
|
|
<table class="subtable">
|
|
<tr>
|
|
<td>FolderGroup:</td><td> $SimularityCalcShareFg</td>
|
|
</tr>
|
|
<tr>
|
|
<td>OwnerFG:</td><td> $SimularityCalcFGOwnerAvg</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Owner:</td><td> $SimularityCalcShareOwner</td>
|
|
</tr>
|
|
<tr>
|
|
<td>MajorityExists:</td><td> $SimularityCalc50P</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Created:</td><td> $SimularityCalcCreateDate</td>
|
|
</tr>
|
|
<tr>
|
|
<td>LastMod:</td><td> $SimularityCalcLastModDate</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<button class="collapsible">$ShareFolderGroupCount</button>
|
|
<div class="content">
|
|
<div class="filelistparent" >
|
|
$ShareFolderGroupList
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<button class="collapsible">$ShareOwnerListCount</button>
|
|
<div class="content">
|
|
<div class="filelistparent" style="font-size: 10px;">
|
|
$ShareOwnerList
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td style="font-size: 10px;">
|
|
$ComputerBar
|
|
$ShareBar
|
|
$AclBar
|
|
</td>
|
|
<td>
|
|
PLACEHOLDER
|
|
</td>
|
|
</tr>
|
|
"@
|
|
$ThisRow
|
|
}
|
|
|
|
# Get share name string list for card
|
|
$CommonShareNamesRollingCount = 0
|
|
$CommonShareNamesTopStringTCard = $CommonShareNamesTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$CommonShareNamesRollingCount = $CommonShareNamesRollingCount + $ShareCount
|
|
$ShareName = $_.name
|
|
$ShareNameBars = Get-GroupNameBar -DataTable $ExcessiveSharePrivs -Name $ShareName -AllComputerCount $ComputerCount -AllShareCount $AllSMBSharesCount -AllAclCount $ShareACLsCount
|
|
$ComputerBar = $ShareNameBars.ComputerBar
|
|
$ShareBar = $ShareNameBars.ShareBar
|
|
$AclBar = $ShareNameBars.AclBar
|
|
$ShareFolderGroupList = $ExcessiveSharePrivs|where sharename -like "$ShareName" | select filelistgroup -Unique | select filelistgroup -ExpandProperty filelistgroup
|
|
$ThisRow = @"
|
|
<tr>
|
|
<td>
|
|
$ShareName
|
|
</td>
|
|
<td>
|
|
$ShareCount
|
|
</td>
|
|
</tr>
|
|
"@
|
|
$ThisRow
|
|
}
|
|
$CommonShareNamesTotalC = [math]::Round($CommonShareNamesRollingCount/$AllSMBSharesCount,4)
|
|
$CommonShareNamesTotalP = $CommonShareNamesTotalC.tostring("P") -replace(" ","")
|
|
|
|
# Get owner string list
|
|
$CommonShareOwnersTop5String = $CommonShareOwnersTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$ShareOwner = $_.name
|
|
Write-Output "$ShareCount $ShareOwner<br>"
|
|
}
|
|
|
|
# Get owner string list table
|
|
$CommonShareOwnersTop5StringT = $CommonShareOwnersTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$ShareOwner = $_.name
|
|
$ShareOwnerBars = Get-GroupOwnerBar -DataTable $ExcessiveSharePrivs -Name $ShareOwner -AllComputerCount $ComputerCount -AllShareCount $AllSMBSharesCount -AllAclCount $ShareACLsCount
|
|
$ComputerBarO = $ShareOwnerBars.ComputerBar
|
|
$ShareBarO = $ShareOwnerBars.ShareBar
|
|
$AclBarO = $ShareOwnerBars.AclBar
|
|
$ThisRow = @"
|
|
<tr>
|
|
<td>
|
|
$ShareCount
|
|
</td>
|
|
<td>
|
|
$ShareOwner
|
|
</td>
|
|
<td>
|
|
$ComputerBarO
|
|
</td>
|
|
<td>
|
|
$ShareBarO
|
|
</td>
|
|
<td>
|
|
$AclBarO
|
|
</td>
|
|
</tr>
|
|
"@
|
|
$ThisRow
|
|
}
|
|
|
|
$NewHtmlReport = @"
|
|
<html>
|
|
<head>
|
|
<link rel="shortcut icon" src="data:image/png;base64, iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyhpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuNi1jMTM4IDc5LjE1OTgyNCwgMjAxNi8wOS8xNC0wMTowOTowMSAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENDIDIwMTcgKE1hY2ludG9zaCkiIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6QTQxQkNBNzA2OEI1MTFFNzlENkRCMzJFODY4RjgwNDMiIHhtcE1NOkRvY3VtZW50SUQ9InhtcC5kaWQ6QTQxQkNBNzE2OEI1MTFFNzlENkRCMzJFODY4RjgwNDMiPiA8eG1wTU06RGVyaXZlZEZyb20gc3RSZWY6aW5zdGFuY2VJRD0ieG1wLmlpZDpBNDFCQ0E2RTY4QjUxMUU3OUQ2REIzMkU4NjhGODA0MyIgc3RSZWY6ZG9jdW1lbnRJRD0ieG1wLmRpZDpBNDFCQ0E2RjY4QjUxMUU3OUQ2REIzMkU4NjhGODA0MyIvPiA8L3JkZjpEZXNjcmlwdGlvbj4gPC9yZGY6UkRGPiA8L3g6eG1wbWV0YT4gPD94cGFja2V0IGVuZD0iciI/Ptdv5vcAAAB9SURBVHjaYmTAAS4IajsCqeVQbqTB+6v7saljxKHZCUhtAWJOqNB3IPYBGrKPoAFYNDPgM4SRSM04DWEkQTNWQxhJ1IxhCCM0tLeSoBnZEG+QAS+ADHEG8sBLJgYKAciASKhzGMjwQiTlgUiVaKRKQqJKUqZKZiI1OwMEGAA7FE70gYsL4wAAAABJRU5ErkJggg==" >
|
|
<title>Report</title>
|
|
<style>
|
|
|
|
.1collapsible:after {
|
|
content: '\0208A';
|
|
font-size: 30;
|
|
color: gray;
|
|
padding: 5px;
|
|
font-weight: bold;
|
|
}
|
|
|
|
.1active:after {
|
|
content: "\0078";
|
|
font-size: 30;
|
|
color: gray;
|
|
padding: 5px;
|
|
--font-weight: bold;
|
|
}
|
|
|
|
.collapsible {
|
|
font-family:"Open Sans", sans-serif;
|
|
font-size:15;
|
|
font-weight:600;
|
|
color: #333;
|
|
padding-left:0px;
|
|
background-color: inherit;
|
|
cursor: pointer;
|
|
border: none;
|
|
outline: none;
|
|
}
|
|
|
|
.active, .collapsible:hover {
|
|
--background-color: #555;
|
|
color:#CE112D;
|
|
--font-weight:bold;
|
|
}
|
|
|
|
.content {
|
|
max-height: 0;
|
|
overflow: hidden;
|
|
transition: max-height 0.2s ease-out;
|
|
}
|
|
|
|
.tabs{
|
|
margin-top: 10px;
|
|
display:-webkit-box;
|
|
display:-ms-flexbox;
|
|
display:flex;
|
|
-ms-flex-wrap:wrap;
|
|
flex-wrap:wrap;
|
|
width:100%
|
|
}
|
|
|
|
.tabInput{
|
|
position:fixed;
|
|
top:0;
|
|
left:0;
|
|
opacity:0
|
|
}
|
|
|
|
.tabLabel{
|
|
width:auto;
|
|
color:#C4C4C8;
|
|
--font-weight:bold;
|
|
--cursor:pointer;
|
|
padding-left: 15px;
|
|
--border-bottom:2px solid transparent;
|
|
-webkit-box-ordinal-group:2;
|
|
-ms-flex-order:1;
|
|
order:1;
|
|
--border-radius:0.80rem 0.80rem 0.80rem 0.80rem
|
|
}
|
|
|
|
.tabLabel:focus{
|
|
outline:0
|
|
background-color:red;
|
|
}
|
|
|
|
.tabInput:target+
|
|
.tabLabel,
|
|
.tabInput:checked+
|
|
.tabLabel{
|
|
--background:white;
|
|
--border-bottom-color:#9B3722;
|
|
--border-bottom:2px solid;
|
|
--color:#9B3722; red
|
|
font-weight: bold;
|
|
}
|
|
|
|
.stuff {
|
|
color:#C4C4C8;
|
|
font-weight: bold;
|
|
font-weight: normal;
|
|
width:auto;
|
|
text-decoration: none;
|
|
padding-top:5px;
|
|
padding-bottom:5px;
|
|
padding-left:15px;
|
|
order:1;
|
|
}
|
|
|
|
.stuff:hover{
|
|
font-weight: normal;
|
|
background-color:#555555;
|
|
text-decoration: none;
|
|
padding-top:5px;
|
|
padding-bottom:5px;
|
|
}
|
|
|
|
|
|
.stuff:active {
|
|
font-weight: normal;
|
|
background-color:#5D5C5C;
|
|
width:auto;
|
|
padding-left: 15px;
|
|
}
|
|
|
|
.stuff:visited {
|
|
font-weight: normal;
|
|
color:#C4C4C8;
|
|
width:auto;
|
|
padding-left: 15px;
|
|
}
|
|
|
|
.tabLabel:hover{
|
|
background-color:#555555;
|
|
color:#ccc;
|
|
--border-color:#eceeef #eceeef #ddd
|
|
}
|
|
|
|
.tabPanel{
|
|
display:none;width:100%;
|
|
-webkit-box-ordinal-group:100;
|
|
-ms-flex-order:99;order:99
|
|
}
|
|
|
|
.tabInput:target+
|
|
.tabLabel+.tabPanel,
|
|
.tabInput:checked+
|
|
.tabLabel+
|
|
.tabPanel{
|
|
display:block
|
|
}
|
|
|
|
.tabPanel.nojs{
|
|
opacity:0
|
|
}
|
|
|
|
{box-sizing:border-box}
|
|
body,html{
|
|
font-family:"Open Sans",
|
|
sans-serif;font-weight:400;
|
|
min-height:100%;
|
|
color:#3d3935;
|
|
margin:0px;
|
|
line-height:1.5;
|
|
overflow-x:hidden
|
|
font-family:"Proxima Nova","Open Sans",-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif;
|
|
font-size:14px;
|
|
font-weight:normal;
|
|
line-height:1.3;
|
|
background-color:#f0f3f5;
|
|
--color:#333;
|
|
--background-color:#DEDFE1
|
|
}
|
|
|
|
table{
|
|
width:100%;
|
|
max-width:100%;
|
|
--margin-bottom:1rem;
|
|
border-collapse:collapse;
|
|
}
|
|
|
|
.tabledrop {
|
|
border-right:1px solid #BEDFE1;
|
|
border-left:1px solid #BEDFE1;
|
|
border-bottom:1px solid #BEDFE1;
|
|
--border:1px solid #757575;
|
|
--border-top:1.5px solid #757575;
|
|
--box-shadow: 0 0 0 0;
|
|
box-shadow: 0 2px 4px 0 #DEDFE1;
|
|
margin: 10px;
|
|
width: 90%;
|
|
--margin-left:10px;
|
|
}
|
|
|
|
table thead th{
|
|
vertical-align:bottom;
|
|
background-color: #3D3935;
|
|
color:white;
|
|
border:1px solid #3D3935;
|
|
}
|
|
|
|
table tbody tr{
|
|
background-color:white;
|
|
--font-weight: bold;
|
|
}
|
|
|
|
table tbody td:nth-child(1){
|
|
--font-weight: bold;
|
|
}
|
|
|
|
table tbody tr:nth-of-type(odd){
|
|
--background-color:#F0E8E8;
|
|
background-color:#f9f9f9;
|
|
}
|
|
|
|
table tbody tr:hover{
|
|
background-color:#ECF1F1;
|
|
--font-weight: bold;
|
|
}
|
|
|
|
table td,table th{
|
|
padding:.75rem;
|
|
line-height:1.5;
|
|
text-align:left;
|
|
font-size:1rem;
|
|
vertical-align:top;
|
|
border-top:1px solid #eceeef
|
|
}
|
|
|
|
.subtable{
|
|
all: unset;
|
|
margin: 0;
|
|
padding: 0;
|
|
border: none;
|
|
background: none;
|
|
color: initial;
|
|
text-align: left;
|
|
font-family:"Proxima Nova","Open Sans",-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif;
|
|
font-size:10px;
|
|
border-collapse: unset;
|
|
}
|
|
|
|
.subtable td {
|
|
background: none;
|
|
font-size:10px;
|
|
text-align: left;
|
|
margin: 0;
|
|
padding: 0;
|
|
border: none;
|
|
border-collapse: unset;
|
|
}
|
|
|
|
.subtable tr {
|
|
background: none;
|
|
font-size:10px;
|
|
text-align: left;
|
|
margin: 0;
|
|
padding: 0;
|
|
border: none;
|
|
border-collapse: unset;
|
|
}
|
|
|
|
.subtable tbody td:nth-child(1) {
|
|
background: none;
|
|
font-size:10px;
|
|
text-align: left;
|
|
margin: 0;
|
|
padding: 0;
|
|
border: none;
|
|
border-collapse: unset;
|
|
}
|
|
|
|
.subtable tbody tr:nth-of-type(odd) {
|
|
background: none;
|
|
font-size:10px;
|
|
text-align: left;
|
|
margin: 0;
|
|
padding: 0;
|
|
border: none;
|
|
border-collapse: unset;
|
|
}
|
|
|
|
.subtable tbody tr:hover {
|
|
background: none;
|
|
font-size:10px;
|
|
text-align: left;
|
|
margin: 0;
|
|
padding: 0;
|
|
border: none;
|
|
border-collapse: unset;
|
|
}
|
|
|
|
h2{
|
|
font-size:2rem
|
|
}
|
|
|
|
h3{
|
|
font-size:1.75rem
|
|
}
|
|
|
|
h4{
|
|
font-size:1.5rem
|
|
}
|
|
|
|
h1,h2,h3,h4,h5,h6{
|
|
margin-bottom:.5rem;
|
|
margin-top:0px;
|
|
font-family:inherit;
|
|
font-weight:500;
|
|
line-height:1.1;
|
|
color:inherit
|
|
}
|
|
|
|
label{
|
|
display:inline-block;
|
|
padding-top:.5rem;
|
|
padding-bottom:.25rem;
|
|
--margin-bottom:.5rem
|
|
}
|
|
|
|
code{
|
|
padding:.2rem .4rem;
|
|
font-size:1rem;
|
|
color:#bd4147;
|
|
background-color:#f7f7f9;
|
|
-border-radius:.25rem
|
|
}
|
|
|
|
p{
|
|
margin-top:0;
|
|
margin-bottom:1rem
|
|
}
|
|
|
|
a,a:visited{
|
|
text-decoration:none;
|
|
font-size: 14;
|
|
color: gray;
|
|
font-weight: bold;
|
|
}
|
|
|
|
a:hover{
|
|
--color:#9B3722;
|
|
text-decoration:underline
|
|
}
|
|
|
|
.preload *{
|
|
-webkit-transition:none !important;
|
|
-moz-transition:none !important;
|
|
-ms-transition:none !important;
|
|
-o-transition:none !important
|
|
}
|
|
|
|
.header{
|
|
text-align:center
|
|
}
|
|
|
|
.noscroll{
|
|
overflow:hidden
|
|
}
|
|
|
|
.link:hover{
|
|
text-decoration:underline
|
|
}
|
|
|
|
li{
|
|
list-style-type:none
|
|
}
|
|
|
|
.mobile{
|
|
display:none;
|
|
height:0;
|
|
width:0
|
|
}
|
|
|
|
@media (max-width: 700px){
|
|
.mobile{display:block !important}
|
|
}
|
|
|
|
code{
|
|
color:black;
|
|
font-family:monospace
|
|
}
|
|
|
|
ul.noindent{
|
|
padding-left:20px
|
|
}
|
|
|
|
.pageDescription {
|
|
margin: 10px;
|
|
width:90%;
|
|
}
|
|
|
|
.pagetitle {
|
|
font-size: 20;
|
|
font-weight:bold;
|
|
--color:#9B3722;
|
|
--color:#CE112D;
|
|
color:#222222;
|
|
}
|
|
|
|
.pagetitlesub {
|
|
font-size: 20;
|
|
font-weight:bold;
|
|
--color:#9B3722;
|
|
color:#CE112D;
|
|
--color:#222222;
|
|
}
|
|
|
|
.topone{background:#999999}
|
|
|
|
.divbarDomain{
|
|
background:#d9d7d7;
|
|
width:200px;
|
|
--border: 1px solid #999999;
|
|
height: 15px;
|
|
text-align:center;
|
|
}
|
|
|
|
.divbarDomainInside{
|
|
--background:#9B3722;
|
|
background:#CE0E2D;
|
|
text-align:center;
|
|
height: 15px;
|
|
vertical-align:middle;
|
|
}
|
|
|
|
.piechartComputers {
|
|
display: block;
|
|
width: 130px;
|
|
height: 130px;
|
|
background: radial-gradient(white 60%, transparent 41%),
|
|
conic-gradient(#CE112D 0% $PercentComputerExPrivP,
|
|
#d9d7d7 $PercentComputerExPrivP 100%);
|
|
border-radius: 50%;
|
|
text-align: center;
|
|
margin-top: 5px;
|
|
margin-bottom: 10px;
|
|
}
|
|
|
|
.piechartShares {
|
|
display: block;
|
|
width: 130px;
|
|
height: 130px;
|
|
background: radial-gradient(white 60%, transparent 41%),
|
|
conic-gradient(#CE112D 0% $PercentSharesExPrivP,
|
|
#d9d7d7 $PercentSharesExPrivP 100%);
|
|
border-radius: 50%;
|
|
text-align: center;
|
|
margin-top: 5px;
|
|
margin-bottom: 10px;
|
|
}
|
|
|
|
.piechartAcls {
|
|
display: block;
|
|
width: 130px;
|
|
height: 130px;
|
|
background: radial-gradient(white 60%, transparent 41%),
|
|
conic-gradient(#CE112D 0% $PercentAclExPrivP,
|
|
#d9d7d7 $PercentAclExPrivP 100%);
|
|
border-radius: 50%;
|
|
text-align: center;
|
|
margin-top: 5px;
|
|
margin-bottom: 10px;
|
|
}
|
|
|
|
.piechartLastAccess {
|
|
display: block;
|
|
width: 130px;
|
|
height: 130px;
|
|
background: radial-gradient(white 60%, transparent 41%),
|
|
conic-gradient(#CE112D 0% $ExpPrivAccessLastP ,
|
|
#d9d7d7 $ExpPrivAccessLastP 100%);
|
|
border-radius: 50%;
|
|
text-align: center;
|
|
margin-top: 5px;
|
|
margin-bottom: 10px;
|
|
}
|
|
|
|
.piechartLastMod {
|
|
display: block;
|
|
width: 130px;
|
|
height: 130px;
|
|
background: radial-gradient(white 60%, transparent 41%),
|
|
conic-gradient(#CE112D 0% $ExpPrivModLastP,
|
|
#d9d7d7 $ExpPrivModLastP 100%);
|
|
border-radius: 50%;
|
|
text-align: center;
|
|
margin-top: 5px;
|
|
margin-bottom: 10px;
|
|
}
|
|
|
|
.percentagetextBuff {
|
|
--height: 25%;
|
|
}
|
|
|
|
.percentagetext {
|
|
text-align: center;
|
|
font-size: 2.25em;
|
|
font-weight: 700;
|
|
font-family:"Open Sans", sans-serif;
|
|
--color:#9B3722;
|
|
color:#CE112D;
|
|
}
|
|
|
|
.percentagetext2 {
|
|
font-size: 10;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
text-align: center;
|
|
}
|
|
|
|
.dashboardsub {
|
|
text-align: center;
|
|
font-size: 12;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
font-weight: bold;
|
|
}
|
|
|
|
.dashboardsub2 {
|
|
font-size: 10;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
text-align: right;
|
|
}
|
|
|
|
.landingheader {
|
|
font-size: 16;
|
|
font-family:"Open Sans", sans-serif;
|
|
color: #CE112D;
|
|
font-weight: bold;
|
|
padding-left:0px;
|
|
}
|
|
|
|
.landingheader2 {
|
|
font-size: 16;
|
|
--font-weight: bold;
|
|
color:White;
|
|
--color:9B3722;
|
|
--background-color: #ccc;
|
|
border-bottom: 2px solid #999;
|
|
padding-left:15px;
|
|
}
|
|
|
|
.landingheader2a {
|
|
background-color: 9B3722;
|
|
--background-color: #999;
|
|
padding-left:120px;;
|
|
padding-right: 5px;
|
|
}
|
|
|
|
.landingheader2b {
|
|
background-color: 9B3722;
|
|
--background-color: #999;
|
|
padding-left: 5px;
|
|
padding-right: 5px;
|
|
margin-top: 10px;
|
|
margin-left: 10px;
|
|
font-size: 16;
|
|
color:White;
|
|
}
|
|
|
|
.landingtext {
|
|
font-size: 14;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
background-color:white;
|
|
border-radius: 10px;
|
|
padding: 20px;
|
|
margin-top: 10px;
|
|
margin-right: 10px;
|
|
margin-bottom: 15px;
|
|
width: 90%
|
|
}
|
|
|
|
.landingtext2 {
|
|
font-size: 14;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
padding-top: 5px;
|
|
padding-bottom: 20px;
|
|
padding-left:15px;
|
|
}
|
|
|
|
.subexpandnocolor {
|
|
font-size: 14;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
background-color:none;
|
|
border-radius: 0px;
|
|
padding: 5px;
|
|
margin-top: 5px;
|
|
margin-right: 5px;
|
|
margin-bottom: 5px;
|
|
width: 90%
|
|
}
|
|
|
|
.filelist {
|
|
font-size: 14;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
background-color:white;
|
|
border-radius: 0px;
|
|
padding: 5px;
|
|
margin-top: 5px;
|
|
margin-right: 5px;
|
|
margin-bottom: 5px;
|
|
width: 90%
|
|
}
|
|
|
|
.filelistparent {
|
|
font-size: 14;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
--background-color:white;
|
|
border-radius: 0px;
|
|
padding: 5px;
|
|
margin-top: 5px;
|
|
margin-right: 5px;
|
|
margin-bottom: 5px;
|
|
width: 90%
|
|
}
|
|
|
|
.tablecolinfo {
|
|
font-size: 14;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
}
|
|
|
|
.card {
|
|
width: 250px;
|
|
box-shadow: 0 2px 4px 0 #DEDFE1;
|
|
transition:0.3s;
|
|
background-color: #3D3935;
|
|
font-family:"Open Sans", sans-serif;
|
|
font-size: 12;
|
|
font-weight: 2;
|
|
font-color: black;
|
|
float: left;
|
|
--overflow:auto;
|
|
display:block;
|
|
margin:10px;
|
|
margin-bottom:20px;
|
|
border-radius: 10px;
|
|
}
|
|
|
|
.card:hover{
|
|
box-shadow: 0 8px 16px 0;
|
|
--box-shadow: 0 8px 16px 0 #DEDFE1;
|
|
|
|
}
|
|
|
|
.cardtitle{
|
|
padding:5px;
|
|
padding-left: 20px;
|
|
font-size: 20;
|
|
color: white;
|
|
font-weight:bold;
|
|
font-family:"Open Sans", sans-serif;
|
|
border-bottom:1.5px solid transparent;
|
|
border-bottom-color:#757575;
|
|
}
|
|
|
|
.cardsubtitle {
|
|
font-size: 12;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#222222;
|
|
text-align: left;
|
|
font-weight: bold;
|
|
}
|
|
|
|
.cardsubtitle2 {
|
|
font-size: 12;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#eee;
|
|
text-align: left;
|
|
font-weight: bold;
|
|
}
|
|
|
|
.cardbartext {
|
|
font-size: 10;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
text-align: right;
|
|
font-weight:bold;
|
|
}
|
|
|
|
.cardbartext2 {
|
|
font-size: 10;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
text-align: right;
|
|
margin-left: 10px;
|
|
}
|
|
|
|
.cardtitlescan{
|
|
padding:5px;
|
|
padding-left: 10px;
|
|
font-size:15;
|
|
color: white;
|
|
font-weight:bold;
|
|
font-family:"Open Sans", sans-serif;
|
|
--border-bottom:1.5px solid transparent;
|
|
--border-bottom-color:#222222;
|
|
background-color: #222222;
|
|
}
|
|
|
|
.cardtitlescansub {
|
|
font-size: 10;
|
|
font-family:"Open Sans", sans-serif;
|
|
color: #eee;
|
|
text-align: center;
|
|
}
|
|
|
|
.cardcontainer {
|
|
background-color:white;
|
|
padding: 8px;
|
|
: center;
|
|
--padding-left: 10px;
|
|
border-right:1px solid #ccc;
|
|
border-left:1px solid #ccc;
|
|
border-bottom:1px solid #ccc;
|
|
border-bottom-right-radius: 10px;
|
|
border-bottom-left-radius: 10px;
|
|
}
|
|
|
|
.cardbarouter{
|
|
background:#d9d7d7;
|
|
width:102px;
|
|
--border: 1px solid #999999;
|
|
height: 15px;
|
|
text-align:center;
|
|
}
|
|
|
|
.cardbarinside{
|
|
--background:#9B3722;
|
|
background:#CE112D;
|
|
text-align:center;
|
|
height: 15px;
|
|
vertical-align:middle;
|
|
width: 0px;
|
|
}
|
|
|
|
.AclEntryWrapper {
|
|
--width: 300px;
|
|
overflow: hidden;
|
|
}
|
|
|
|
.AclEntryLeft {
|
|
width: 75px;
|
|
float:left;
|
|
font-size: 12;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
--font-weight:bold;
|
|
}
|
|
.AclEntryRight{
|
|
float: left;
|
|
font-size: 12;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#666;
|
|
}
|
|
|
|
.sidenavcred{
|
|
font-size: 12;
|
|
font-family:"Open Sans", sans-serif;
|
|
color: white;
|
|
text-align: left;
|
|
padding-left:15px;
|
|
}
|
|
|
|
.sidenav {
|
|
box-shadow: 0 2px 4px 0;
|
|
width: 180px;
|
|
height: 100%;
|
|
background-color:#222222;
|
|
position: fixed; /* Stay in place */
|
|
top: 0;
|
|
left: 0;
|
|
float:left;
|
|
line-height:1.15;
|
|
-webkit-text-size-adjust:100%;
|
|
-ms-text-size-adjust:100%;
|
|
}
|
|
|
|
.sidenav a {
|
|
width:auto;
|
|
cursor:initial;
|
|
border-bottom:2px solid transparent;
|
|
-webkit-box-ordinal-group:2;
|
|
-ms-flex-order:1;
|
|
order:1;
|
|
}
|
|
|
|
#main {
|
|
margin-left: 190px;
|
|
margin-right: 10px;
|
|
--padding-left: 20px;
|
|
}
|
|
|
|
.Minicard {
|
|
width: 115px;
|
|
box-shadow: 0 2px 4px 0 #DEDFE1;
|
|
transition:0.3s;
|
|
background-color: #BDBDBD;
|
|
font-family:"Open Sans", sans-serif;
|
|
font-size: 12;
|
|
font-weight: 2;
|
|
font-color: black;
|
|
float: left;
|
|
--overflow:auto;
|
|
display:block;
|
|
margin:0px;
|
|
margin-bottom:20px;
|
|
border-radius: 10px;
|
|
}
|
|
|
|
.Minicard:hover{
|
|
--box-shadow: 0 8px 16px 0;
|
|
box-shadow: 0 8px 16px 0 #CDCECF;
|
|
}
|
|
|
|
.Minicardtitle{
|
|
padding:5px;
|
|
--padding-left: 20px;
|
|
font-size: 13;
|
|
color: #222222;
|
|
font-weight:bold;
|
|
font-family:"Open Sans", sans-serif;
|
|
border-bottom:1.5px solid transparent;
|
|
border-bottom-color:#757575;
|
|
}
|
|
|
|
.Minicardcontainer {
|
|
background-color:white;
|
|
padding: 8px;
|
|
: center;
|
|
--padding-left: 10px;
|
|
border-right:1px solid #ccc;
|
|
border-left:1px solid #ccc;
|
|
border-bottom:1px solid #ccc;
|
|
border-bottom-right-radius: 10px;
|
|
border-bottom-left-radius: 10px;
|
|
}
|
|
|
|
.MinicardconnectLine {
|
|
display: block;
|
|
float:left;
|
|
border-bottom: 1px solid red;
|
|
height:10px;
|
|
width:20px;
|
|
margin-top:58px;
|
|
}
|
|
.TimelineChart{
|
|
display: grid;
|
|
--grid-template-columns: 1px repeat($ExcessivePrivsYearsCount, 204px) 1px;
|
|
grid-template-rows: minmax(0px, 1fr);
|
|
overflow-x: scroll;
|
|
overflow-y: hidden;
|
|
scroll-snap-type: x proximity;
|
|
margin:5px;
|
|
border:1px solid #ccc;
|
|
background-color: #F2F3F4;
|
|
}
|
|
|
|
.TimelineChart:before,
|
|
.TimelineChart:after {
|
|
content: '';
|
|
}
|
|
|
|
.TimelineChart> li,
|
|
.YearItem {
|
|
display: flex;
|
|
scroll-snap-align: left;
|
|
padding:0;
|
|
margin: 0;
|
|
display: block;
|
|
flex-direction: row;
|
|
justify-content: left;
|
|
align-items: left;
|
|
background: #F0F2F4;
|
|
}
|
|
|
|
.TimelineBarOutside {
|
|
position: relative;
|
|
height: 153px;
|
|
width: 15px;
|
|
float: left;
|
|
display: block;
|
|
bottom: 0;
|
|
left: 0;
|
|
}
|
|
|
|
.TimelineBarOutside:hover{
|
|
background-color: white;
|
|
|
|
}
|
|
|
|
.TimelineBarOutside:hover > div{
|
|
opacity: 1;
|
|
}
|
|
|
|
.TimelineBarInside:hover > .TimelinePopup{
|
|
visibility:visible;
|
|
}
|
|
|
|
.PopWrapper{
|
|
}
|
|
|
|
.PopWrapper:hover > .TimelinePopup {
|
|
visibility:visible;
|
|
z-index: 6;
|
|
|
|
}
|
|
|
|
.TimelineBarInside {
|
|
width: 100%;
|
|
display: block;
|
|
background-color: #CE112D;
|
|
opacity: .5;
|
|
margin-left:1px;
|
|
position: absolute;
|
|
top: 100%;
|
|
-ms-transform: translateY(-100%);
|
|
transform: translateY(-100%);
|
|
}
|
|
|
|
.TimelineDot {
|
|
width: 100%;
|
|
height:15px;
|
|
display: block;
|
|
background-color: blue;
|
|
position: absolute;
|
|
bottom: 0;
|
|
left: 0;
|
|
opacity: .5;
|
|
}
|
|
|
|
.TimelinePopup {
|
|
|
|
visibility:hidden;
|
|
}
|
|
|
|
.TimelineMinicard {
|
|
width: 115px;
|
|
box-shadow: 0 2px 4px 0 #DEDFE1;
|
|
background-color: #BDBDBD;
|
|
font-family:"Open Sans", sans-serif;
|
|
font-size: 10;
|
|
font-weight: 2;
|
|
font-color: black;
|
|
float: left;
|
|
display:block;
|
|
margin:0px;
|
|
margin-left: 20px;
|
|
margin-top: 10px;
|
|
-margin-bottom:20px;
|
|
border-radius: 10px;
|
|
}
|
|
|
|
.TimelineMinicard:hover{
|
|
box-shadow: 0 8px 16px 0 #CDCECF;
|
|
}
|
|
|
|
.TimelineMinicardtitle{
|
|
padding:5px;
|
|
font-size: 10;
|
|
color: #222222;
|
|
font-weight:bold;
|
|
font-family:"Open Sans", sans-serif;
|
|
border-bottom:1.5px solid transparent;
|
|
border-bottom-color:#757575;
|
|
}
|
|
|
|
.TimelineMinicardcontainer {
|
|
background-color:white;
|
|
padding: 8px;
|
|
border-right:1px solid #ccc;
|
|
border-left:1px solid #ccc;
|
|
border-bottom:1px solid #ccc;
|
|
border-bottom-right-radius: 10px;
|
|
border-bottom-left-radius: 10px;
|
|
}
|
|
|
|
.TimelineMinicardconnectLine {
|
|
display: block;
|
|
float:left;
|
|
border-bottom: 1px solid red;
|
|
height:10px;
|
|
width:20px;
|
|
margin-top:58px;
|
|
}
|
|
|
|
.LargeCard {
|
|
width: 788px;
|
|
box-shadow: 0 2px 4px 0 #DEDFE1;
|
|
transition:0.3s;
|
|
background-color: #3D3935;
|
|
font-family:"Open Sans", sans-serif;
|
|
font-size: 12;
|
|
font-weight: 2;
|
|
font-color: black;
|
|
float: left;
|
|
display:block;
|
|
margin:10px;
|
|
margin-bottom:20px;
|
|
border-radius: 10px;
|
|
}
|
|
|
|
.LargeCard:hover{
|
|
box-shadow: 0 8px 16px 0;
|
|
}
|
|
|
|
.LargeCardtitle{
|
|
padding:5px;
|
|
padding-left: 20px;
|
|
font-size: 20;
|
|
color: white;
|
|
font-weight:bold;
|
|
font-family:"Open Sans", sans-serif;
|
|
border-bottom:1.5px solid transparent;
|
|
border-bottom-color:#757575;
|
|
}
|
|
|
|
.LargeCardSubtitle2 {
|
|
font-size: 12;
|
|
font-family:"Open Sans", sans-serif;
|
|
color:#eee;
|
|
text-align: left;
|
|
font-weight: bold;
|
|
}
|
|
|
|
.LargeCardContainer {
|
|
background-color:white;
|
|
padding: 8px;
|
|
border-right:1px solid #ccc;
|
|
border-left:1px solid #ccc;
|
|
border-bottom:1px solid #ccc;
|
|
border-bottom-right-radius: 10px;
|
|
border-bottom-left-radius: 10px;
|
|
}
|
|
|
|
</style>
|
|
</head>
|
|
<body onload="radiobtn = document.getElementById('home');radiobtn.checked = true;">
|
|
|
|
<!--
|
|
|||||||||| SIDE MENU
|
|
-->
|
|
<div class="sidenav">
|
|
|
|
<div style="border-bottom:1px solid red;width: 80%;display: block;margin-left:20px">
|
|
<div style="font-size: 15;font-weight:bolder;color:white;margin-bottom:5px; margin-top:28px;" align="center">
|
|
POWERHUNT
|
|
SHARES
|
|
</div>
|
|
</div>
|
|
|
|
<div style="width: 80%;display: block;margin-left:20px">
|
|
<div style="font-size: 15;font-weight:bolder;color:white;margin-bottom:10px; margin-top:10px;text-align:center;">$TargetDomain</div>
|
|
</div>
|
|
|
|
<div id="tabs" class="tabs" data-tabs-ignore-url="false">
|
|
<label href="#" class="stuff" style="width:100%;margin-top:15px" onClick="radiobtn = document.getElementById('home');radiobtn.checked = true;">Home</label>
|
|
<label class="tabLabel" style="width:100%;color:white;background-color:#333;border-top:1px solid #757575;padding-top:5px;padding-bottom:5px;margin-top:1px;margin-bottom:2px;font-weight:bolder"><Strong>Reports</Strong></label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('dashboard');radiobtn.checked = true;">Dashboard</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('computersummary');radiobtn.checked = true;">Computer Summary</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('sharesum');radiobtn.checked = true;">Share Summary</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('ACLsum');radiobtn.checked = true;">ACL Summary</label>
|
|
<label class="tabLabel" style="width:100%;color:white;background-color:#333;padding-top:5px;padding-bottom:5px;margin-top:2px;margin-bottom:2px;"><Strong>Data Insights</Strong></label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('accounts');radiobtn.checked = true;">Group Stats</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('ShareName');radiobtn.checked = true;">Top Share Names</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('ShareOwner');radiobtn.checked = true;">Top Share Owners</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('ShareFolders');radiobtn.checked = true;">Top Folder Groups</label>
|
|
<label href="#" class="stuff" style="width:100%;" onclick="radiobtn = document.getElementById('SubNets');radiobtn.checked = true;">Affected Subnets</label>
|
|
<label class="tabLabel" style="width:100%;color:white;background-color:#333;padding-top:5px;padding-bottom:5px;margin-top:2px;margin-bottom:2px;"><strong>Recommendations</strong></label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('Attacks');radiobtn.checked = true;">Exploit Share Access</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('Detections');radiobtn.checked = true;">Detect Share Scans</label>
|
|
<label href="#" class="stuff" style="width:100%;" onClick="radiobtn = document.getElementById('Remediation');radiobtn.checked = true;">Prioritize Remediation</label>
|
|
</div>
|
|
<img style="float: right;" src="data:image/png;base64, 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" />
|
|
<div class="sidenavcred" style="border-top:1px solid #757575;position: absolute;bottom:0px;width:91%;color:#757575;background-color:#222222">
|
|
<br>
|
|
<strong style="color:#DEDFE1;">Invoke-HuntSMBShares</strong><br>
|
|
<strong>Author:</strong> Scott Sutherland<br>
|
|
<strong>License:</strong> 3-clause BSD<br>
|
|
<br>
|
|
</div>
|
|
</div>
|
|
<div id="main">
|
|
|
|
<!--
|
|
|||||||||| PAGE: Dashboard
|
|
-->
|
|
<input class="tabInput" name="tabs" type="radio" id="dashboard"/>
|
|
<label class="tabLabel" onClick="updateTab('dashboard',false)" for="dashboard"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="min-height: 450px;margin-top:11px;">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Dashboard Charts</h2>
|
|
Below is a summary of the shares configured with excessive privileges on accessible domain computers.
|
|
<a href="$SubDir/$ShareACLsExFile">Download Details</a>
|
|
</div>
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<!--
|
|
|||||||||| CARD: COMPUTER SUMMARY
|
|
-->
|
|
|
|
<div style="margin-left: 10px;">$ExcessiveSharePrivsCount ACL entries, on $ExcessiveSharesCount shares, hosted by $ComputerWithExcessive computers were found configured with excessive privilegs on the $TargetDomain domain.</div>
|
|
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('computersummary');radiobtn.checked = true;">
|
|
<div class="card">
|
|
<div class="cardtitle">
|
|
Computers<br>
|
|
<span class="cardsubtitle2">host shares with excessive privileges</span>
|
|
</div>
|
|
<div class="cardcontainer" align="center" style="padding-bottom: 22px;">
|
|
<span class="piechartComputers">
|
|
<span class="percentagetext">
|
|
<div class="percentagetextBuff"></div>
|
|
<img style ="padding-top:20px; padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;" src="data:image/png;base64, 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Oi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvbW0vIgogICAgeG1sbnM6c3RFdnQ9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZUV2ZW50IyIKICAgIHhtbG5zOmRjPSJodHRwOi8vcHVybC5vcmcvZGMvZWxlbWVudHMvMS4xLyIKICAgIHhtbG5zOkdJTVA9Imh0dHA6Ly93d3cuZ2ltcC5vcmcveG1wLyIKICAgIHhtbG5zOnRpZmY9Imh0dHA6Ly9ucy5hZG9iZS5jb20vdGlmZi8xLjAvIgogICAgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIgogICB4bXBNTTpEb2N1bWVudElEPSJnaW1wOmRvY2lkOmdpbXA6NjBhMGY0NjUtNjJmYS00ZmVlLTk5NGUtNmYwYzc5ZmIwODhkIgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOmM2NmZkYzhhLTgxMjMtNDBiYS1iYTNiLTg2YTIwNDE2MGIxMSIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOmE1MjRlOTA5LWU1MjEtNGZiMC05MmMyLTExMzZjOWU1NjE1ZCIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMjM3NzY2Mjc4NDAwIgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6OTFjYWZiZWEtOTFiZi00YzEwLTliMzMtM2I0MWYyZmQ4NTE3IgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI2VDE2OjU2OjA2Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/PiNxLrcAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWOAboWgp4AAADyklEQVRIx61WX2hbZRT/ne+7N1mWtUnadLrYdquxoHQES1qx0251DJniH8YUfRN8ENGXPU4QwRcffHIviogvCiKCA1FhwjbBB6FL7FilnYRRS6hNejfnTdLb/L3f8aE39SZL0lvwwIXLd757fuec3/l936VTp+YI/5n7vWncy18uV2AYBgUCAT54cIg7fA8NvY27OVZWVmU2u5awbfs0gBEARSHo11AodDmRmLA0bSc0kasS2gWAAICZsbBwo980Cx8w82sAfACU6/uUrutvzcw89rvPpwMAhOP0BAAAmcwt3TQLF5j5DQAZInpT0+RTQoizAC4CmK7X69/Mz6cPM2+HoUgk1G9Z5fsApu1YDMcHInLwCERQUkqrUqkmlVLfElE6ENh3Npl8dMPv94OZsbz8hy+fN95l5neI6PPx8fjbo6PDigD8DCDplNyVGyJaBHAFQJyZXxVCPHfixJNXpBQtVafTCyHTLF4DEAqHQw9PTU0WNACTRPSRlOIHZjBtp99pAFgIYdRqta8BVA8cCF5rBwBA0eigVSiU0sx8xrKsGABTc9qyMjc3m+oxqgwAi4tLPsO4XQcgbNv2ddpv24qYeZ+zXG8SjyZBu01ZNDrQIKLrAPzlcuVZy7Ja9jIzcrmN+wEcA/BXONy/vgPi6lDPKYvFDikpxZcAtpRS76dS16ez2TXJzDDNAs3Pp6PVauUCgCEi+ioWO1QGQBLAeSK6NDZ2eNGjEDdMs1BlxotKqVfu3v3nodXVbHx9Pf98rVb7EMATTvKiWCxdGh6ObbpBbnhRejgcZqVUulgs/cnMUwCOM/PTAGYA6ET0I4AHAIw3Go3pfN74iQCYQohzJ08e/2I3ADdvpVIJmcytvmJxM8msRgAqCEGmbasXiCillPoYQAjAZc3rWdXOWV9fH5LJyZKjMwDA1au/vM7MLw8MRN4zTbNq2+pTABGxB4CO2mlbEwAgpeBE4uh3fr9vav/+wGnNY0Daq39wcEDNzh7L7hz1zCyXlm7qO+kIAaUUhBCQUnYMopTi5l6XSbfPfZ8IZj6Xy228tGs51FlKLi2POKfBPZcWATjqPL1JYE/U5doTEQBs/L9m67rWgqIR0SfM/EwPcvcydUxE3weDwWpLm8fGRrVSaVPTdb0F5M6dvycaDftMcyzbeZdSXhwaGly2bQWlbAYAXdcpGAzWjxwZbemOFo8/aLtaRi5hPc7M57uBKKXWJiYeWfAy/lq3WReC0sz0WTcQIeg3r/pq/q3cw8fWVhmGcZu2SaQWeogIkUhYBQIB9iBQ/hfq8Z63CHuDFwAAAABJRU5ErkJggg==" />
|
|
$PercentComputerExPrivP
|
|
<span class="percentagetext2" style="margin-top:0px;display: block;">
|
|
<span style="color:#9B3722;font-size:12;">$ComputerWithExcessive</span>
|
|
of $ComputerCount
|
|
</span>
|
|
</span>
|
|
</span>
|
|
<table>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align: top; width:78px;">
|
|
Read Access
|
|
$CheckStatusComputerR
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentComputerReadP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentComputerReadP <br> $ComputerWithReadCount of $ComputerCount</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">
|
|
Write Access
|
|
$CheckStatusComputerW
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentComputerWriteP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentComputerWriteP <br> $ComputerWithWriteCount of $ComputerCount</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">
|
|
High Risk
|
|
$CheckStatusComputerH
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentComputerHighRiskP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentComputerHighRiskP <br> $ComputerwithHighRisk of $ComputerCount</span>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
|
|
<!--
|
|
|||||||||| CARD: SHARE SUMMARY
|
|
-->
|
|
|
|
<a href="#" id="ShareLink" onClick="radiobtn = document.getElementById('sharesum');radiobtn.checked = true;">
|
|
<div class="card">
|
|
<div class="cardtitle">
|
|
Shares<br>
|
|
<span class="cardsubtitle2">configured with excessive privileges</span>
|
|
</div>
|
|
<div class="cardcontainer" align="center" style="padding-bottom: 22px;">
|
|
<span class="piechartShares">
|
|
<span class="percentagetext">
|
|
<div class="percentagetextBuff"></div>
|
|
<img style ="padding-top:20px; padding-bottom:5px;border-bottom:1px solid #ccc;padding-left:10px; padding-right:10px;" src="data:image/png;base64, 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Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZUV2ZW50IyIKICAgIHhtbG5zOmRjPSJodHRwOi8vcHVybC5vcmcvZGMvZWxlbWVudHMvMS4xLyIKICAgIHhtbG5zOkdJTVA9Imh0dHA6Ly93d3cuZ2ltcC5vcmcveG1wLyIKICAgIHhtbG5zOnRpZmY9Imh0dHA6Ly9ucy5hZG9iZS5jb20vdGlmZi8xLjAvIgogICAgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIgogICB4bXBNTTpEb2N1bWVudElEPSJnaW1wOmRvY2lkOmdpbXA6YzczMGM1Y2ItYjRjMi00YmRhLThlMDAtMWI3MDI1NjMwMDBiIgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOmEzNGQyZDc5LTMxZjMtNDYwYy1hM2M2LTFiNjFiNjljZTMzMyIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOjNiZmI0Y2M5LWE0YWQtNDNjMy04ZTkwLTE1MDljNzkxNmU2NCIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMjM1NjI3NzQ0MTc0IgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6NjNjM2VmYTAtNmI3Yi00NGZhLWFlZjctY2EzMjJlYzA2OGRjIgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI2VDE2OjIwOjI3Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/PrSHjn8AAAAGYktHRADoAPQASsUjROMAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWFBuybQ0PAAABi0lEQVRIx+2Tv0vDQBTH3zVpKsUghCb2x2C4lBq4paClSwfBzCVDQVBxDO3m6tBJl+zdBHERlwz9E1ycrHSyOkibDo6FDqaLtudiixcVpd4i9MF3eXfw4XPvHsCiFvXvy5Fl+R4AWuEoitLBGDcqlcrqnwjJZLLjuu7Sd+e5XG5fkqTbUql0nMlk4vNyWj9d8DxPIIQcmqb5+JXxNNls9sayrO25IL+ter2updPpu3AfvUM2pw3Lspbb7fZJEARxWZZRNBoFSikAAFBKmYzHY5hMJrNQSmE0GtmSJDUFQYBYLAbFYvHhk0mhUDgCAMozkbDaYDDY4/l1CSFXDMS2bdLtdglPSCqVOmcgvV5vlydAUZRnURQ9Zia6rvs8Z4ExPgMAmJnk8/kN3/fXeJqYpnnJ7IlhGC5PC8MwnqaAmQlC6ICnRb/fP2U2vlwub/HejVqtts5AMMYNngBd168/AiIIIRgOhzs8n0pV1YtwL+BpIYpi4DjOCkPQNK2qadoLD4Cqqq+JRKIatngD35X+UDOXLeMAAAAASUVORK5CYII=" />
|
|
$PercentSharesExPrivP
|
|
<span class="percentagetext2" style="margin-top:1px;display: block;">
|
|
<span style="color:#9B3722;font-size:12;">$ExcessiveSharesCount</span>
|
|
of $AllSMBSharesCount
|
|
</span>
|
|
</span>
|
|
</span>
|
|
<table>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align: top; width:78px;">
|
|
Read Access
|
|
$CheckStatusShareR
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentSharesReadP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentSharesReadP <br> $SharesWithReadCount of $AllSMBSharesCount</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">
|
|
Write Access
|
|
$CheckStatusShareW
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentSharesWriteP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentSharesWriteP <br> $SharesWithWriteCount of $AllSMBSharesCount</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">
|
|
High Risk
|
|
$CheckStatusShareH
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentSharesHighRiskP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentSharesHighRiskP <br> $SharesHighRiskCount of $AllSMBSharesCount</span>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
|
|
<!--
|
|
|||||||||| CARD: ACL SUMMARY
|
|
-->
|
|
|
|
<a href="#" id="AclLink" onClick="radiobtn = document.getElementById('ACLsum');radiobtn.checked = true;">
|
|
<div class="card">
|
|
<div class="cardtitle">
|
|
Share ACLs<br>
|
|
<span class="cardsubtitle2">configured with excessive privileges</span>
|
|
</div>
|
|
<div class="cardcontainer" align="center" style="padding-bottom: 22px;">
|
|
<span class="piechartAcls">
|
|
<span class="percentagetext">
|
|
<div class="percentagetextBuff"></div>
|
|
<img style ="padding-top:20px; padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;" src="data:image/png;base64, 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MC00M2FhLTk3MmEtMzM2MDEyNzQzY2RjIgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOjU4MTg5M2UwLWFmZTktNDhkNi1hN2VlLThhMTU3MjA2MGQzMiIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOjBlMjA5NmU2LTNmYWYtNGEzOS1hMjQyLWMxNDFmMWU4MDQzYiIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMjM2OTc2MzY1NjM0IgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6MDYwMjZhYTEtZWQyYy00NDU2LWExZTAtYzdlZWVjZThiNDY1IgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI2VDE2OjQyOjU2Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/PvHyhAIAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWKjhRz2cAAAAA8klEQVRIx+3WMUoDQRTG8d9m13SpLMQLWNiqbTB4CAs7EcFKEI8hKdIpKfUWQhCxUtRK9AAWsbMRBMnabMBiSLLrrI15MDB8fDP/ebw3zDCPEpHgAEfIp3jfsI3XKqAXNGfwbeIRy1UgTyW8HdxjqQwgK3mgAY5xjY8JvrzwHCJPikxWsYiViLU+wSnOx5A2rnATubFaSMc12UKvpg5+bhSTFKOaIKPGX1zGOeSfQrJfrt/AekC/w20syFcxQnq0TNawF7qAeIgFucBlQB/GzGQH+wH9DP1YkP7PzSa1cIaFGd74KtFEmqCLXXzivYaPSvcbYO4pyzEBUV8AAAAASUVORK5CYII=" />
|
|
$PercentAclExPrivP
|
|
<span class="percentagetext2" style="margin-top:1px;display: block;">
|
|
<span style="color:#9B3722;font-size:12;">$ExcessiveSharePrivsCount</span>
|
|
of $ShareACLsCount
|
|
</span>
|
|
</span>
|
|
</span>
|
|
<table>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align: top; width:78px;">
|
|
Read Access
|
|
$CheckStatusAclR
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentAclReadP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentAclReadP <br> $AclWithReadCount of $ShareACLsCount</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">
|
|
Write Access
|
|
$CheckStatusAclw
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentAclWriteP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentAclWriteP <br> $AclWithWriteCount of $ShareACLsCount</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">
|
|
High Risk
|
|
$CheckStatusAclH
|
|
</td>
|
|
<td align="right">
|
|
<div class="cardbarouter">
|
|
<div class="cardbarinside" style="width: $PercentAclHighRiskP;"></div>
|
|
</div>
|
|
<span class="cardbartext">$PercentAclHighRiskP <br> $AclHighRiskCount of $ShareACLsCount</span>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
|
|
<!--
|
|
|||||||||| CARD: Top 5 Names
|
|
-->
|
|
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('ShareName');radiobtn.checked = true;">
|
|
<div class="card">
|
|
<div class="cardtitle">
|
|
Top Share Names<br>
|
|
<span class="cardsubtitle2">configured with excessive privileges</span>
|
|
</div>
|
|
<div class="cardcontainer" align="center">
|
|
<span class="piechartComputers">
|
|
<span class="percentagetext">
|
|
<div class="percentagetextBuff"></div>
|
|
<img style ="padding-top:20px; padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;" src="data:image/png;base64, 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LTRhMGEtOWY2Yi01MjZlNjNjNzJmYWYiCiAgIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6MDY0N2QwMjQtN2Y0ZC00MzU0LThkZDYtMjA0ODY3ZGFiMjczIgogICB4bXBNTTpPcmlnaW5hbERvY3VtZW50SUQ9InhtcC5kaWQ6OGI5OWVjZmMtYmI4NC00YWYwLTg2NGEtZTVmM2E0NGRkYzVlIgogICBkYzpGb3JtYXQ9ImltYWdlL3BuZyIKICAgR0lNUDpBUEk9IjIuMCIKICAgR0lNUDpQbGF0Zm9ybT0iV2luZG93cyIKICAgR0lNUDpUaW1lU3RhbXA9IjE2NDQ0MjAyOTYyMDQ4NjMiCiAgIEdJTVA6VmVyc2lvbj0iMi4xMC4yOCIKICAgdGlmZjpPcmllbnRhdGlvbj0iMSIKICAgeG1wOkNyZWF0b3JUb29sPSJHSU1QIDIuMTAiPgogICA8eG1wTU06SGlzdG9yeT4KICAgIDxyZGY6U2VxPgogICAgIDxyZGY6bGkKICAgICAgc3RFdnQ6YWN0aW9uPSJzYXZlZCIKICAgICAgc3RFdnQ6Y2hhbmdlZD0iLyIKICAgICAgc3RFdnQ6aW5zdGFuY2VJRD0ieG1wLmlpZDo2YWViZTRjNy05YjU1LTQ4NGQtOTUzZS0xMWViODgxYWM4YWUiCiAgICAgIHN0RXZ0OnNvZnR3YXJlQWdlbnQ9IkdpbXAgMi4xMCAoV2luZG93cykiCiAgICAgIHN0RXZ0OndoZW49IjIwMjItMDItMDlUMDk6MjQ6NTYiLz4KICAgIDwvcmRmOlNlcT4KICAgPC94bXBNTTpIaXN0b3J5PgogIDwvcmRmOkRlc2NyaXB0aW9uPgogPC9yZGY6UkRGPgo8L3g6eG1wbWV0YT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgIAo8P3hwYWNrZXQgZW5kPSJ3Ij8+VWGc2AAAAAZiS0dEAOgA9ABKxSNE4wAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB+YCCQ8YOKpEKS8AAAGFSURBVEjH3dYxSJVRFAfw3+u9HNQwjJe4CUGEIIqOQSg1ZKGLSEPbo8UpqJzcdFFQh5bQURRpaSkdiyhti1yktpaWSEkIQRRfy3nwcNDv6bWhb7lw7jnf/97z/59zLv/Ll8MInqB8gu9PPMCP0wB9Q10Gv158QetpQDZr8O3DZ7TUAlCo8UDv8AwfsHuMXzl8HqOci5u04wquJ+R6GnNYrIDcwnusJRbWJeQrnNzG83NS8NcKJ3kcZgi4h7vYwyt8yhBzeKGGEy3Fj6+hG+uYyBKYFeQRBjCJG2jGOMZwM5WEh7CMUdSHrRErsbeW4iaXg4f6KlsRv9GQKl1vcQcLVbZ53MfHVOmawUN0hNRzGA75L6cC2cZTDKIf+1jFSxykAingRfSk2WiQpVBWVypOBoLoqzEavqMJnehJBVKK9RfeRGHuH9k7U7oaQ0HrkaYDbOF11EkxhFA+C8gfTEWNtFXZx7GRta0UcDHDjN89MkWzANQhnwu1lKKid87hoTL7T55EfwHPxkx+CTMteQAAAABJRU5ErkJggg==" />
|
|
$CommonShareNamesTotalP
|
|
<span class="percentagetext2" style="margin-top:0px;display: block;">
|
|
<span style="color:#9B3722;font-size:12;">$CommonShareNamesRollingCount</span>
|
|
of $AllSMBSharesCount
|
|
</span>
|
|
</span>
|
|
</span>
|
|
<table>
|
|
$CommonShareNamesTopStringTCard
|
|
</table>
|
|
</div>
|
|
</div>
|
|
<div style="height:.5px;width:100%;position:relative;float:left;"></div>
|
|
|
|
|
|
<!--
|
|
|||||||||| CARD: Share Creation Summary
|
|
-->
|
|
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('DashBoard');radiobtn.checked = true;">
|
|
<div class="card">
|
|
<div class="cardtitle">
|
|
Shares Created<br>
|
|
<span class="cardsubtitle2">in last $ShareCreationDays days</span>
|
|
</div>
|
|
<div class="cardcontainer" align="center">
|
|
<span class="piechartComputers">
|
|
<span class="percentagetext">
|
|
<div class="percentagetextBuff"></div>
|
|
<img style ="padding-top:20px; padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;" src="data:image/png;base64, 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LTQzZjQtYTU2OC1hZTNiZGUyODZhOTUiCiAgIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6YTQwMDdjMDktZjM2YS00ZjEzLTg0MTEtMDg0YzY1ZWIyMmQyIgogICB4bXBNTTpPcmlnaW5hbERvY3VtZW50SUQ9InhtcC5kaWQ6MmE5NmQwYmItODFhMS00N2VmLTlhZmMtMzRmNzBhODM0YTMxIgogICBkYzpGb3JtYXQ9ImltYWdlL3BuZyIKICAgR0lNUDpBUEk9IjIuMCIKICAgR0lNUDpQbGF0Zm9ybT0iV2luZG93cyIKICAgR0lNUDpUaW1lU3RhbXA9IjE2NDQ0MjA0ODgyMjk0ODYiCiAgIEdJTVA6VmVyc2lvbj0iMi4xMC4yOCIKICAgdGlmZjpPcmllbnRhdGlvbj0iMSIKICAgeG1wOkNyZWF0b3JUb29sPSJHSU1QIDIuMTAiPgogICA8eG1wTU06SGlzdG9yeT4KICAgIDxyZGY6U2VxPgogICAgIDxyZGY6bGkKICAgICAgc3RFdnQ6YWN0aW9uPSJzYXZlZCIKICAgICAgc3RFdnQ6Y2hhbmdlZD0iLyIKICAgICAgc3RFdnQ6aW5zdGFuY2VJRD0ieG1wLmlpZDo4ZTdiNmU2ZS05NzhjLTQ3ZDMtYmIwZS0yZTc1OGExMDJjOWUiCiAgICAgIHN0RXZ0OnNvZnR3YXJlQWdlbnQ9IkdpbXAgMi4xMCAoV2luZG93cykiCiAgICAgIHN0RXZ0OndoZW49IjIwMjItMDItMDlUMDk6MTU6NTIiLz4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6YmFlNTE3ZWEtMGU5Ni00NzZjLThkYWUtNDMyMGE4OWQwOGZlIgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAyLTA5VDA5OjI4OjA4Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/PvnuaGIAAAAGYktHRADoAPQASsUjROMAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmAgkPHAjo8dyHAAAA5ElEQVRIx+3VvUoDURCG4WfzIwlEG7G0sUxja2OdKoU3I96ClTaWegFprL2EqNiKrV1IShGFuGszCxJw48Ix1X7wdcO8Z344wwaUYR9j9CviCrzgDsu6kDYmeMUC77/4Ayc4wD3yuqDHNVWUGuAaZ+jWbdcDjuO1fwFdROxzRVweeZ9QtFZa11vjJU4xxVaFB7jEEFknAH2c4zDhUu3hBlcl5CjoY3wlBO3itoRsY4a3WNdUyldn8m9qIA2kgTSQxOoEqPfjnqT8IHfQyuLSjSL5PHERbUyyqKYbpzi1CnxuYiS+AUTqLN3dBmCqAAAAAElFTkSuQmCC" />
|
|
$ExpPrivCreationLastP
|
|
<span class="percentagetext2" style="margin-top:0px;display: block;">
|
|
<span style="color:#9B3722;font-size:12;">$ExPrivCreationLastnShareCount</span>
|
|
of $AllSMBSharesCount
|
|
</span>
|
|
</span>
|
|
</span>
|
|
<table>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align: top; width:78px;">Computers</td>
|
|
<td align="right">
|
|
$ExPrivCreationLastComputerB
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Shares</td>
|
|
<td align="right">
|
|
$ExPrivCreationLastShareB
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">ACLs</td>
|
|
<td align="right">
|
|
$ExPrivCreationLastShareAclB
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
|
|
<!--
|
|
|||||||||| CARD: Share Creation Timeline
|
|
-->
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('DashBoard');radiobtn.checked = true;">
|
|
$CardCreationTimeLine
|
|
</a>
|
|
<div style="height:.5px;width:100%;position:relative;float:left;"></div>
|
|
|
|
<!--
|
|
|||||||||| CARD: Last Access Summary
|
|
-->
|
|
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('DashBoard');radiobtn.checked = true;">
|
|
<div class="card">
|
|
<div class="cardtitle">
|
|
Shares Accessed<br>
|
|
<span class="cardsubtitle2">in last $LastAccessDays days</span>
|
|
</div>
|
|
<div class="cardcontainer" align="center">
|
|
<span class="piechartComputers">
|
|
<span class="percentagetext">
|
|
<div class="percentagetextBuff"></div>
|
|
<img style ="padding-top:20px; padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;" src="data:image/png;base64, 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NGI4NS04M2NhLTRmYThhYTFhODAyYSIKICAgeG1wTU06SW5zdGFuY2VJRD0ieG1wLmlpZDpiZjU2YjU0YS1lNjNiLTRlNjktYjIwMS02YTE1OTZiMjBjMzUiCiAgIHhtcE1NOk9yaWdpbmFsRG9jdW1lbnRJRD0ieG1wLmRpZDpmNDNiOWQ5YS00ZDJhLTRjYmItOTYxYS03YjBmMmQ2YjU2NWMiCiAgIGRjOkZvcm1hdD0iaW1hZ2UvcG5nIgogICBHSU1QOkFQST0iMi4wIgogICBHSU1QOlBsYXRmb3JtPSJXaW5kb3dzIgogICBHSU1QOlRpbWVTdGFtcD0iMTY0NDQyMDQzNzQ3MDU5OSIKICAgR0lNUDpWZXJzaW9uPSIyLjEwLjI4IgogICB0aWZmOk9yaWVudGF0aW9uPSIxIgogICB4bXA6Q3JlYXRvclRvb2w9IkdJTVAgMi4xMCI+CiAgIDx4bXBNTTpIaXN0b3J5PgogICAgPHJkZjpTZXE+CiAgICAgPHJkZjpsaQogICAgICBzdEV2dDphY3Rpb249InNhdmVkIgogICAgICBzdEV2dDpjaGFuZ2VkPSIvIgogICAgICBzdEV2dDppbnN0YW5jZUlEPSJ4bXAuaWlkOjBjZjc0ZTQ5LTM4MGEtNGQ0Zi04NzQ0LWJkNjFmYzYzMzc3ZiIKICAgICAgc3RFdnQ6c29mdHdhcmVBZ2VudD0iR2ltcCAyLjEwIChXaW5kb3dzKSIKICAgICAgc3RFdnQ6d2hlbj0iMjAyMi0wMi0wOVQwOToyNzoxNyIvPgogICAgPC9yZGY6U2VxPgogICA8L3htcE1NOkhpc3Rvcnk+CiAgPC9yZGY6RGVzY3JpcHRpb24+CiA8L3JkZjpSREY+CjwveDp4bXBtZXRhPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgCjw/eHBhY2tldCBlbmQ9InciPz4HW/wsAAAABmJLR0QA6AD0AErFI0TjAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH5gIJDxsRw9vigAAAAcZJREFUSMfd1U+IzVEUB/CPZwZlKNmNUkYkG8xYaEyajSxEWcziTZFQUmPBgoWk/CkbywklOxtKEfmz0AylZsKGnZKFP2VhSuYpyrM5r65rfu+PN6s5dfrde/7c7z1/7vkxl2gYn/FjBv6KwXYB5uEjzmNiBv0AzmEr3rQDUsE2vAjZaaxLbLbjF8ZaPPsSXkFHphjFfnTiaMie/sflj6G3BiIi2Ywz+IYTUY926AkO1TalRNGLLlyMRqilbmesb2IVFmM8ZH24FusDEcE/lKfrCK5GTTahH4vwKeo2kKw3xrc/bAfjAmNYgpVYg7dpuu7gMG6hit+zwFVcySNZi11YHm+kXerDRA5SxoMCgE5sSPbvmrjIFjzP01XFUIFDd+hrXG4iknGM5JF8x/1ZGlcrYlIMlTLFvYgspYV4jIeZ/AJeYnUBSDney5c0Xe+xu8BhX5aqGl+uE8kkDqYvfgRTWFDH6VEG8AFLC2x7Yt4tS0Ge4XqDHPdgOgHZU8f2FO6mgko47WiimCfD9nYDu9d591WiOKUmQOZHW3bXsVkfZ3blIKMttGZHA/3ZGE1/OUzhZ0zhdqmEvTjeyj++VZ7GjUjrHKQ/8x6Qeu2jMHoAAAAASUVORK5CYII=" />
|
|
$ExpPrivAccessLastP
|
|
<span class="percentagetext2" style="margin-top:0px;display: block;">
|
|
<span style="color:#9B3722;font-size:12;">$ExPrivAccessLastnShareCount</span>
|
|
of $AllSMBSharesCount
|
|
</span>
|
|
</span>
|
|
</span>
|
|
<table>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align: top; width:78px;">Computers</td>
|
|
<td align="right">
|
|
$ExPrivAccesLastComputerB
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Shares</td>
|
|
<td align="right">
|
|
$ExPrivAccesLastShareB
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">ACLs</td>
|
|
<td align="right">
|
|
$ExPrivAccesLastShareAclB
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
|
|
<!--
|
|
|||||||||| CARD: LastAccessDate Timeline
|
|
-->
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('DashBoard');radiobtn.checked = true;">
|
|
$CardLastAccessTimeLine
|
|
</a>
|
|
<div style="height:.5px;width:100%;position:relative;float:left;"></div>
|
|
|
|
|
|
<!--
|
|
|||||||||| CARD: Last Modified Summary
|
|
-->
|
|
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('DashBoard');radiobtn.checked = true;">
|
|
<div class="card">
|
|
<div class="cardtitle">
|
|
Shares Modified<br>
|
|
<span class="cardsubtitle2">in last $LastModDays days</span>
|
|
</div>
|
|
<div class="cardcontainer" align="center">
|
|
<span class="piechartComputers">
|
|
<span class="percentagetext">
|
|
<div class="percentagetextBuff"></div>
|
|
<img style ="padding-top:20px; padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;" src="data:image/png;base64, iVBORw0KGgoAAAANSUhEUgAAABkAAAAZCAYAAADE6YVjAAAC8XpUWHRSYXcgcHJvZmlsZSB0eXBlIGV4aWYAAHja7ZdNktwgDIX3nCJHsCSExHEwmKrcIMfPk+12pn9m0ckqVQ1lwAI/hD6gZ9L26+dMP5CoeklZzUstZUHKNVduaPhypLaXtOS9PF7q2Uf39nR1MEyCWo5XL+f4m50ugaNqaOkXIe9nx3rfUfOp7w9CfFQSHkV7nEL1FBI+OugUaMeyllLdvi5h3Y76/P4IA54URfZ7t5/eDdEbinmEeROSBSWLHw5IPJyk7Y0oKwaiRJuloBShUwwBeRWnKyHOaYar+eWgOypXi17b0yOtzOcQeQhyueqX9kT60CHXPPx15uxni+/tJodUWh6iH8+cw+e+Zqyi5YJQl3NRt6XsLYxbMUVM7Ql6ZTE8Cgnbc0V27OqOrTCWvqzInSoxcE3KNKjRpG2vO3W4mHlLbGgwd5bd6GJcuYMeSY5Mkw0khzhYdmAXWPnyhfZp69LTPptj5kEYygQxin3xbk7vfjBnHAWixa9YwS/mCDbcCHJRYhiI0DyDqnuAb/kxBVcBQY0oxxGpCOx6SKxKf24C2UELBirq4wySjVMAIcLUCmdIQADUSJQKLcZsRAikA1CD6yyZVxAgVR5wkrNIARvnmBqfGO1DWRnmBDsuM5BQnC8DG5w1wMpZsX8sO/ZQU9GsqkVNXau2IiUXLaVYiUuxmVhOplbMzK1ac/Hs6sXN3au3ylVwaWot1arXWlvDnA3KDV83DGht5VXWvGpay2qrr3VtHdun5669dOvea2+DhwzcH6MMGz7qaBtt2Epb3nQrm22+1a1NbLUpaeaps0ybPutsF7UT61N+gxqd1HgnFQPtogar2U2C4jrRYAZgnDKBuAUCbGgOZotTzhzkgtlSGadCGU5qMBsUxEAwb8Q66cYu8UE0yP0Tt2T5jhv/LbkU6N4k98ztFbURP0N9J3acwgjqIjh96N+8sbf4sXuq03cd79YfoY/QR+gj9BH6CH2E/iOhiT8e4r/A34qgp6W3nZ5SAAABhWlDQ1BJQ0MgcHJvZmlsZQAAeJx9kT1Iw0AcxV9TpSpVBzuIOGSoThZERQQXqWIRLJS2QqsOJpd+QZOGJMXFUXAtOPixWHVwcdbVwVUQBD9AHJ2cFF2kxP8lhRYxHhz34929x907QKiXmWp2jAOqZhnJWFTMZFfFwCsEdKMPswhLzNTjqcU0PMfXPXx8vYvwLO9zf45eJWcywCcSzzHdsIg3iKc3LZ3zPnGIFSWF+Jx4zKALEj9yXXb5jXPBYYFnhox0cp44RCwW2lhuY1Y0VOIp4rCiapQvZFxWOG9xVstV1rwnf2Ewp62kuE5zGDEsIY4ERMioooQyLERo1UgxkaT9qId/yPEnyCWTqwRGjgVUoEJy/OB/8LtbMz854SYFo0Dni21/jACBXaBRs+3vY9tunAD+Z+BKa/krdWDmk/RaSwsfAf3bwMV1S5P3gMsdYPBJlwzJkfw0hXweeD+jb8oCA7dAz5rbW3Mfpw9AmrpavgEODoHRAmWve7y7q723f880+/sBuspyxJYT6vsAAA0YaVRYdFhNTDpjb20uYWRvYmUueG1wAAAAAAA8P3hwYWNrZXQgYmVnaW49Iu+7vyIgaWQ9Ilc1TTBNcENlaGlIenJlU3pOVGN6a2M5ZCI/Pgo8eDp4bXBtZXRhIHhtbG5zOng9ImFkb2JlOm5zOm1ldGEvIiB4OnhtcHRrPSJYTVAgQ29yZSA0LjQuMC1FeGl2MiI+CiA8cmRmOlJERiB4bWxuczpyZGY9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkvMDIvMjItcmRmLXN5bnRheC1ucyMiPgogIDxyZGY6RGVzY3JpcHRpb24gcmRmOmFib3V0PSIiCiAgICB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIKICAgIHhtbG5zOnN0RXZ0PSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvc1R5cGUvUmVzb3VyY2VFdmVudCMiCiAgICB4bWxuczpkYz0iaHR0cDovL3B1cmwub3JnL2RjL2VsZW1lbnRzLzEuMS8iCiAgICB4bWxuczpHSU1QPSJodHRwOi8vd3d3LmdpbXAub3JnL3htcC8iCiAgICB4bWxuczp0aWZmPSJodHRwOi8vbnMuYWRvYmUuY29tL3RpZmYvMS4wLyIKICAgIHhtbG5zOnhtcD0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wLyIKICAgeG1wTU06RG9jdW1lbnRJRD0iZ2ltcDpkb2NpZDpnaW1wOjE2MTYxYmY4LTI4YzgtNDAxZC1iYjY4LTJjNzgyNzA3ZDRmZiIKICAgeG1wTU06SW5zdGFuY2VJRD0ieG1wLmlpZDo1NzE0ZjExMy01NWY2LTRiYzEtYjE4NS02YWQ2NjFjMzgyZjYiCiAgIHhtcE1NOk9yaWdpbmFsRG9jdW1lbnRJRD0ieG1wLmRpZDpiODBlNGVmYi0zNTM5LTQ5MTUtOGQzZS0yMjU4OTMyMjZiMGMiCiAgIGRjOkZvcm1hdD0iaW1hZ2UvcG5nIgogICBHSU1QOkFQST0iMi4wIgogICBHSU1QOlBsYXRmb3JtPSJXaW5kb3dzIgogICBHSU1QOlRpbWVTdGFtcD0iMTY0NDQyMDM5MjMxODQ4NiIKICAgR0lNUDpWZXJzaW9uPSIyLjEwLjI4IgogICB0aWZmOk9yaWVudGF0aW9uPSIxIgogICB4bXA6Q3JlYXRvclRvb2w9IkdJTVAgMi4xMCI+CiAgIDx4bXBNTTpIaXN0b3J5PgogICAgPHJkZjpTZXE+CiAgICAgPHJkZjpsaQogICAgICBzdEV2dDphY3Rpb249InNhdmVkIgogICAgICBzdEV2dDpjaGFuZ2VkPSIvIgogICAgICBzdEV2dDppbnN0YW5jZUlEPSJ4bXAuaWlkOmFhZWFkNDk3LTkxZmItNGRlNS1iYjQzLWQ4OWZjODI1ZTQzMyIKICAgICAgc3RFdnQ6c29mdHdhcmVBZ2VudD0iR2ltcCAyLjEwIChXaW5kb3dzKSIKICAgICAgc3RFdnQ6d2hlbj0iMjAyMi0wMi0wOVQwOToyNjozMiIvPgogICAgPC9yZGY6U2VxPgogICA8L3htcE1NOkhpc3Rvcnk+CiAgPC9yZGY6RGVzY3JpcHRpb24+CiA8L3JkZjpSREY+CjwveDp4bXBtZXRhPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgCjw/eHBhY2tldCBlbmQ9InciPz6Pz8otAAAABmJLR0QA6AD0AErFI0TjAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH5gIJDxogix7T+wAAActJREFUSMfd1U2ITmEUB/DfvIbko0hNRFOIZCFmLDQmzUYWYodGKflIaixY2EkZCykrTY2yZWEWk6hZKA0LNZINNmxsfEQRMvSS1+a8eua673u97qzmX6f73POcc//P/zzneS4zCfvxBt9y7AP6yhK04RXOYyJnvheD2IqnZUgmsQ2PwncG65KY7fiB8Ra/fQmPoT0zMYSDmI0T4bv/H4s/ia46iVCyGWfxGadjP8rgLo7UXyrJRBcW4EI0Qr10O2N8AysxH/fC142rMT4UCv5CtlzHcSX2ZBN6MBevY996k/HGePZEbF8sYBwL0Yk1eJGW6yaOYQQ1/JoGq2E4q2QtdmFJnJGy6MZEJePsx1gTgjasaIFkCx5kSZbhepOkAVxrgWRvNMyUPfmCeQ0SOmO+hqP/QLAcP9GRVXI7SPNwOVocLmJpAUl/nJd3qZKX2N0gYU8oSG2kgOQhDqcnfgAfMScneBHe5pDUohPzsCruu8Xpid+HUVQbdMgoniW+TxjGhialGouF/1FSw44C+UOJgucFsU+CaMq18h53ChKrSVN8bxK3HqtxK3VOxiqnC+eyTdEedavGLVwWFRzAqVb+8a3a17gRZpmR+A0U8pApMCYr0AAAAABJRU5ErkJggg==" />
|
|
$ExpPrivModifiedLastP
|
|
<span class="percentagetext2" style="margin-top:0px;display: block;">
|
|
<span style="color:#9B3722;font-size:12;">$ExPrivModifiedLastnShareCount</span>
|
|
of $AllSMBSharesCount
|
|
</span>
|
|
</span>
|
|
</span>
|
|
<table>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align: top; width:78px;">Computers</td>
|
|
<td align="right">
|
|
$ExPrivModifiedLastComputerB
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Shares</td>
|
|
<td align="right">
|
|
$ExPrivModifiedLastShareB
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">ACLs</td>
|
|
<td align="right">
|
|
$ExPrivModifiedLastShareAclB
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
|
|
<!--
|
|
|||||||||| CARD: LastModifiedDate Timeline
|
|
-->
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('DashBoard');radiobtn.checked = true;">
|
|
$CardLastModifiedTimeLine
|
|
</a>
|
|
|
|
</div>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: COMPUTER SUMMARY
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="computersummary"/>
|
|
<label class="tabLabel" onClick="updateTab('computersummary',false)" for="computersummary"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Computer Summary</h2>
|
|
Below is a summary of the domain computers that were targeted, connectivity to them, and the number that are hosting potentially insecure SMB shares.
|
|
</div>
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th>Description</th>
|
|
<th align="left">Percent Chart</th>
|
|
<th align="left">Percent</th>
|
|
<th align="left">Computers</th>
|
|
<th align="left">Details</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr>
|
|
<td>DISCOVERED</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: 100%;"></div></div></td>
|
|
<td>100.00%</td>
|
|
<td>$ComputerCount</td>
|
|
<td><a href="$SubDir/$DomainComputersFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$DomainComputersFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>PING RESPONSE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentComputerPingP;"></div></div></td>
|
|
<td>$PercentComputerPingP</td>
|
|
<td>$ComputerPingableCount</td>
|
|
<td><a href="$SubDir/$ComputersPingableFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ComputersPingableFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>PORT 445 OPEN</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentComputerPortP;"></div></div></td>
|
|
<td>$PercentComputerPortP</td>
|
|
<td>$Computers445OpenCount</td>
|
|
<td><a href="$SubDir/$Computers445OpenFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$Computers445OpenFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HOST SHARE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentComputerWitShareP;"></div></div></td>
|
|
<td>$PercentComputerWitShareP</td>
|
|
<td>$AllComputersWithSharesCount</td>
|
|
<td><a href="$SubDir/$ShareACLsFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HOST NON-DEFAULT SHARE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentComputerNonDefaultP;"></div></div></td>
|
|
<td>$PercentComputerNonDefaultP</td>
|
|
<td>$ComputerwithNonDefaultCount</td>
|
|
<td><a href="$SubDir/$ShareACLsNonDefaultFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsNonDefaultFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HOST POTENTIALLY INSECURE SHARE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width:$PercentComputerExPrivP;"></div></div></td>
|
|
<td>$PercentComputerExPrivP</td>
|
|
<td>$ComputerWithExcessive</td>
|
|
<td><a href="$SubDir/$ShareACLsExFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsExFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HOST READABLE SHARE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentComputerReadP;"></div></div></td>
|
|
<td>$PercentComputerReadP</td>
|
|
<td>$ComputerWithReadCount</td>
|
|
<td><a href="results/$ShareACLsReadFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsReadFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HOST WRITEABLE SHARE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentComputerWriteP;"></div></div></td>
|
|
<td>$PercentComputerWriteP</td>
|
|
<td>$ComputerWithWriteCount</td>
|
|
<td><a href="$SubDir/$ShareACLsWriteFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsWriteFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HOST HIGH RISK SHARE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentComputerHighRiskP;"></div></div></td>
|
|
<td>$PercentComputerHighRiskP</td>
|
|
<td>$ComputerwithHighRisk</td>
|
|
<td><a href="$SubDir/$ShareACLsHRFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsHRFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: SHARE SUMMARY
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="sharesum"/>
|
|
<label class="tabLabel" onClick="updateTab('sharesum,false)" for="sharesum"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Share Summary</h2>
|
|
Below is a summary of the SMB shares discovered on domain computers that may provide excessive privileges to standard domain users.
|
|
</div>
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th>Description</th>
|
|
<th align="left">Percent Chart</th>
|
|
<th align="left">Percent</th>
|
|
<th align="left">Shares</th>
|
|
<th align="left">Details</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr>
|
|
<td>DISCOVERED</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: 100%;"></div></div></td>
|
|
<td>100.00%</td>
|
|
<td>$AllSMBSharesCount</td>
|
|
<td><a href="$SubDir/$AllSMBSharesFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$AllSMBSharesFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>NON-DEFAULT</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentSharesNonDefaultP;"></div></div></td>
|
|
<td>$PercentSharesNonDefaultP</td>
|
|
<td>$SharesNonDefaultCount</td>
|
|
<td><a href="$SubDir/$ShareACLsNonDefaultFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsNonDefaultFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>POTENTIALLY EXCESSIVE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentSharesExPrivP;"></div></div></td>
|
|
<td>$PercentSharesExPrivP</td>
|
|
<td>$ExcessiveSharesCount</td>
|
|
<td><a href="$SubDir/$ShareACLsExFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsExFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>READ ACCESS</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentSharesReadP;"></div></div></td>
|
|
<td>$PercentSharesReadP</td>
|
|
<td>$SharesWithReadCount</td>
|
|
<td><a href="$SubDir/$ShareACLsReadFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsReadFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>WRITE ACCESS</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentSharesWriteP;"></div></div></td>
|
|
<td>$PercentSharesWriteP</td>
|
|
<td>$SharesWithWriteCount</td>
|
|
<td><a href="$SubDir/$ShareACLsWriteFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsWriteFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HIGH RISK</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentSharesHighRiskP;"></div></div></td>
|
|
<td>$PercentSharesHighRiskP</td>
|
|
<td>$SharesHighRiskCount</td>
|
|
<td><a href="$SubDir/$ShareACLsHRFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsHRFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
<div class="pageDescription">
|
|
Note: All Windows systems have a c$ and admin$ share configured by default. A a result, the number of visible shares should be (at a minimum) double the number of the computers found with port 445 open. In this case, $Computers445OpenCount computers were found with port 445 open, so we would expect to discover approximetly $MinExpectedShareCount or more shares.
|
|
</div>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: ACL SUMMARY
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="ACLsum"/>
|
|
<label class="tabLabel" onClick="updateTab('ACLsum',false)" for="ACLsum"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Share ACL Entry Summary</h2>
|
|
Below is a summary of the SMB share ACL entries discovered on domain computers that may provide excessive privileges to standard domain users.
|
|
</div>
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th>Description</th>
|
|
<th align="left">Percent Chart</th>
|
|
<th align="left">Percent</th>
|
|
<th align="left">ACLs</th>
|
|
<th align="left">Details</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr>
|
|
<td>DISCOVERED</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: 100%;"></div></div></td>
|
|
<td>100.00%</td>
|
|
<td>$ShareACLsCount</td>
|
|
<td><a href="$SubDir/$ShareACLsFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>NON-DEFAULT</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentAclNonDefaultP;"></div></div></td>
|
|
<td>$PercentAclNonDefaultP</td>
|
|
<td>$AclNonDefaultCount</td>
|
|
<td><a href="$SubDir/$ShareACLsNonDefaultFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsNonDefaultFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>POTENTIALLY EXCESSIVE</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentAclExPrivP;"></div></div></td>
|
|
<td>$PercentAclExPrivP</td>
|
|
<td>$ExcessiveSharePrivsCount</td>
|
|
<td><a href="$SubDir/$ShareACLsExFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsExFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>READ ACCESS</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentAclReadP;"></div></div></td>
|
|
<td>$PercentAclReadP</td>
|
|
<td>$AclWithReadCount</td>
|
|
<td><a href="$SubDir/$ShareACLsReadFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsReadFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>WRITE ACCESS</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentAclWriteP;"></div></div></td>
|
|
<td>$PercentAclWriteP</td>
|
|
<td>$AclWithWriteCount</td>
|
|
<td><a href="$SubDir/$ShareACLsWriteFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsWriteFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
<tr>
|
|
<td>HIGH RISK</td>
|
|
<td><div class="divbarDomain"><div class="divbarDomainInside" style="width: $PercentAclHighRiskP;"></div></div></td>
|
|
<td>$PercentAclHighRiskP</td>
|
|
<td>$AclHighRiskCount</td>
|
|
<td><a href="$SubDir/$ShareACLsHRFile"><span class="cardsubtitle">CSV</a> | </span><a href="$SubDir/$ShareACLsHRFileH"><span class="cardsubtitle">HTML</span></a></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: GROUP STATS
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="accounts"/>
|
|
<label class="tabLabel" onClick="updateTab('accounts',false)" for="accounts"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Data Insights: Group Stats</h2>
|
|
In the context of this report, excessive read and write share permissions have been defined as any network share ACL containing an explicit entry for the "Everyone", "Authenticated Users", "BUILTIN\Users", "Domain Users", or "Domain Computers" groups. All provide domain users access to the affected shares due to privilege inheritance.
|
|
Below is a summary of the exposure associated with each of those groups.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th align="left">Name</th>
|
|
<th align="left">Excessive ACL Entries</th>
|
|
<th align="left">Affected Computers</th>
|
|
<th align="left">Affected Shares</th>
|
|
<th align="left">Affected ACLs</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr>
|
|
<td>Everyone</td>
|
|
<td>
|
|
<span class="tablecolinfo">
|
|
<div class="AclEntryWrapper">
|
|
<div class="AclEntryLeft">
|
|
Read<br>
|
|
Write<br>
|
|
High Risk<br>
|
|
</div>
|
|
<div class="AclEntryRight">
|
|
: $AceEveryoneAclReadCount <br>
|
|
: $AceEveryoneAclWriteCount <br>
|
|
: $AceEveryoneAclHRCount
|
|
</div>
|
|
</div>
|
|
</span>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceEveryoneComputerCountPS ($AceEveryoneComputerCount of $ComputerCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceEveryoneComputerCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceEveryoneShareCountPS ($AceEveryoneShareCount of $AllSMBSharesCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceEveryoneShareCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceEveryoneAclPS ($AceEveryoneAclCount of $ShareACLsCount)</span>
|
|
<br>
|
|
<div class="divbarDomain"><div class="divbarDomainInside" style="width: $AceEveryoneAclPS;"></div></div>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>BUILTIN\Users</td>
|
|
<td>
|
|
<span class="tablecolinfo">
|
|
<div class="AclEntryWrapper">
|
|
<div class="AclEntryLeft">
|
|
Read<br>
|
|
Write<br>
|
|
High Risk<br>
|
|
</div>
|
|
<div class="AclEntryRight">
|
|
: $AceUsersAclReadCount <br>
|
|
: $AceUsersAclWriteCount <br>
|
|
: $AceUsersAclHRCount
|
|
</div>
|
|
</div>
|
|
</span>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceUsersComputerCountPS ($AceUsersComputerCount of $ComputerCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceUsersComputerCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceUsersShareCountPS ($AceUsersShareCount of $AllSMBSharesCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceUsersShareCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceUsersAclPS ($AceUsersAclCount of $ShareACLsCount)</span>
|
|
<br>
|
|
<div class="divbarDomain"><div class="divbarDomainInside" style="width: $AceUsersAclPS;"></div></div>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Authenticated Users</td>
|
|
<td>
|
|
<span class="tablecolinfo">
|
|
<div class="AclEntryWrapper">
|
|
<div class="AclEntryLeft">
|
|
Read<br>
|
|
Write<br>
|
|
High Risk<br>
|
|
</div>
|
|
<div class="AclEntryRight">
|
|
: $AceAuthenticatedUsersAclReadCount <br>
|
|
: $AceAuthenticatedUsersAclWriteCount <br>
|
|
: $AceAuthenticatedUsersAclHRCount
|
|
</div>
|
|
</div>
|
|
</span>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceAuthenticatedUsersComputerCountPS ($AceAuthenticatedUsersComputerCount of $ComputerCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceAuthenticatedUsersComputerCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceAuthenticatedUsersShareCountPS ($AceAuthenticatedUsersShareCount of $AllSMBSharesCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceAuthenticatedUsersShareCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceAuthenticatedUsersAclPS ($AceAuthenticatedUsersAclCount of $ShareACLsCount)</span>
|
|
<br>
|
|
<div class="divbarDomain"><div class="divbarDomainInside" style="width: $AceAuthenticatedUsersAclPS;"></div></div>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Domain Users</td>
|
|
<td>
|
|
<span class="tablecolinfo">
|
|
<div class="AclEntryWrapper">
|
|
<div class="AclEntryLeft">
|
|
Read<br>
|
|
Write<br>
|
|
High Risk<br>
|
|
</div>
|
|
<div class="AclEntryRight">
|
|
: $AceDomainUsersAclReadCount <br>
|
|
: $AceDomainUsersAclWriteCount <br>
|
|
: $AceDomainUsersAclHRCount
|
|
</div>
|
|
</div>
|
|
</span>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceDomainUsersComputerCountPS ($AceDomainUsersComputerCount of $ComputerCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceDomainUsersComputerCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceDomainUsersShareCountPS ($AceDomainUsersShareCount of $AllSMBSharesCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceDomainUsersShareCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceDomainUsersAclPS ($AceDomainUsersAclCount of $ShareACLsCount)</span>
|
|
<br>
|
|
<div class="divbarDomain"><div class="divbarDomainInside" style="width: $AceDomainUsersAclPS;"></div></div>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Domain Computers</td>
|
|
<td>
|
|
<span class="tablecolinfo">
|
|
<div class="AclEntryWrapper">
|
|
<div class="AclEntryLeft">
|
|
Read<br>
|
|
Write<br>
|
|
High Risk<br>
|
|
</div>
|
|
<div class="AclEntryRight">
|
|
: $AceDomainComputersAclReadCount <br>
|
|
: $AceDomainComputersAclWriteCount <br>
|
|
: $AceDomainComputersAclHRCount
|
|
</div>
|
|
</div>
|
|
</span>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceDomainComputersComputerCountPS ($AceDomainComputersComputerCount of $ComputerCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceDomainComputersComputerCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceDomainComputersShareCountPS ($AceDomainComputersShareCount of $AllSMBSharesCount)</span>
|
|
<br>
|
|
<div class="divbarDomain">
|
|
<div class="divbarDomainInside" style="width: $AceDomainComputersShareCountPS;"></div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<span class="dashboardsub2">$AceDomainComputersAclPS ($AceDomainComputersAclCount of $ShareACLsCount)</span>
|
|
<br>
|
|
<div class="divbarDomain"><div class="divbarDomainInside" style="width: $AceDomainComputersAclPS;"></div></div>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: SHARE NAMES
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="ShareName"/>
|
|
<label class="tabLabel" onClick="updateTab('ShareName',false)" for="ShareName"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Data Insights: $SampleSum Most Common Share Names</h2>
|
|
This section contains a list of the most common SMB share names. In some cases, shares with the exact same name may be related to a single application or process. This information can help identify the root cause associated with the excessive privileges and expedite remediation.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th align="left">Shares</th>
|
|
<th align="left">Share Name</th>
|
|
<th align="left">Similarity</th>
|
|
<th align="left">Folder Groups</th>
|
|
<th align="left">Share Owners</th>
|
|
<th align="left">Affected Assets</th>
|
|
<th align="left">Timeline</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
$CommonShareNamesTopStringT
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: Affected Subnets
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="SubNets">
|
|
<label class="tabLabel" onclick="updateTab('SubNets#39;,false)" for="SubNets"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Data Insights: Affected Subnets</h2>
|
|
This section contains a list of subnets hosting computers with shares that are configured with accessibe privileges.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
$SubnetSummaryHTML
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: SHARE OWNERS
|
|
-->
|
|
<input class="tabInput" name="tabs" type="radio" id="ShareOwner"/>
|
|
<label class="tabLabel" onClick="updateTab('ShareOwner',false)" for="ShareOwner"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Data Insights: $SampleSum Most Common Share Owners</h2>
|
|
This section lists the most common share owners.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th align="left">Share Count</th>
|
|
<th align="left">Owner</th>
|
|
<th align="left">Affected Computers</th>
|
|
<th align="left">Affected Shares</th>
|
|
<th align="left">Affected ACLs</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
$CommonShareOwnersTop5StringT
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: SHARE FOLDERS
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="ShareFolders"/>
|
|
<label class="tabLabel" onClick="updateTab('ShareFolders',false)" for="ShareFolders"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Data Insights: $SampleSum Most Common Share Folder Groups</h2>
|
|
Folder groups are SMB shares that contain the exact same file listing. Each file group has been hashed so they can be quickly correlated. In some cases, shares with the exact same file listing may be related to a single application or process. This information can help identify the root cause associated with the excessive privileges and expedite remediation.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th align="left">Affected Shares</th>
|
|
<th align="left">File Group</th>
|
|
<th align="left">File Count</th>
|
|
<th align="left">Affected Computers</th>
|
|
<th align="left">Affected Shares</th>
|
|
<th align="left">Affected ACLs</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
$CommonShareFileGroupTopString
|
|
</tbody>
|
|
</table>
|
|
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: Exploit Shares
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="Attacks"/>
|
|
<label class="tabLabel" onClick="updateTab('Attacks',false)" for="Attacks"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:7px">
|
|
<h3>Exploit Share Accesss</h3>
|
|
Below are some tips for getting started on exploiting share access.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th align="left">Share</th>
|
|
<th align="left">Access</th>
|
|
<th align="left">Instructions</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr>
|
|
<td>C$, admin$</td>
|
|
<td>READ</td>
|
|
<td>
|
|
Read OS and Application password files and log in.<br>
|
|
Identify non-public information disclosure.
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>C$, admin$</td>
|
|
<td>WRITE</td>
|
|
<td>Read OS and Application password files and log in.<br>
|
|
Identify non-public information disclosure.<br>
|
|
Execute arbitrary code by writing files to autorun locations:<br>
|
|
DLL Hijacking<br>
|
|
All Users folders<br>
|
|
Other file based autoruns<br>
|
|
EXE Replacement<br>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>wwwroot,inetpub,webroot</td>
|
|
<td>READ</td>
|
|
<td>Read connection strings and escalation through database. <br>
|
|
<span class="code">Code - search for file types</span><br>
|
|
<span class="code">Code - search for file contents</span><br>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>wwwroot,inetpub,webroot</td>
|
|
<td>Write</td>
|
|
<td>
|
|
Read connection strings and escalation through database.<br>
|
|
Upload webshell to execute as web server service account.
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: Detect Share Scans
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="Detections"/>
|
|
<label class="tabLabel" onClick="updateTab('Detections',false)" for="Detections"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Recommendations: Exploit Share Access</h2>
|
|
Below are some tips for getting started on building detections for potentially malicious share scanning events.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th align="left">Action</th>
|
|
<th align="left">Detection Guidance</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr>
|
|
<td>Detect Share Scanning</td>
|
|
<td>
|
|
<strong>Data Sources</strong><br>
|
|
Ensure that group policy audit settings are configured so that authentication successes and failures are logged so that real-time analysis and offline analysis can be used to identify common indicators of compromise. Specifically, ensure the following events IDs are logged and forward to a SIEM solutions.
|
|
<br><br>
|
|
Logon Success<br>
|
|
- Windows Server 2003: 540<br>
|
|
- Windows Server 2008-2012: 4624<br>
|
|
<br>
|
|
Logon Failure<br>
|
|
- Windows Server 2003: 680<br>
|
|
- Windows Server 2008-2012: 4625<br>
|
|
<br>
|
|
Network Share Object was Accessed
|
|
- All versions: 5140
|
|
|
|
<br><br>
|
|
<strong>Detection Thresholds and Indicators</strong>
|
|
<br>
|
|
Below is a list of common Indicators of Compromise (IoCs) that can be used to identify potentially SMB scanning. Please note that not all IoCs will work in every environment due false positives generated by legitimate applications and processes. However, in some environments it may be possible to modify IoC thresholds or signatures to reduce the number of false positives to an acceptable level.
|
|
<br><br>
|
|
Consider creating correlation rules based on Active Directory and Local Windows authentication logs for:
|
|
<br>
|
|
A single system authenticates to many systems via SMB (port 445) in short period of time, and accesses Windows shares. For example, ten or more systems in under a minute. Use the events above to build detections.
|
|
<br><br>
|
|
Consider implementing a honey pot or canary system that supports SMB shares that can be used to generate alerts when accessed.
|
|
<br><br>
|
|
<strong>Prevention</strong><br>
|
|
If network shares are not required, disable them or block access using host-based firewalls.
|
|
Ensure that sensitive information is not available on these shares. To restrict access under Windows, open Explorer, right-click on each of the shares, go to the 'Sharing' tab, and click on 'Permissions'. From here, add or remove permissions for various users and groups.
|
|
Guest access to the system should also be revoked and ensure that adequate access controls are in place for each shared resource. NULL sessions should be disabled.
|
|
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Detect Canaries</td>
|
|
<td>Build detections for authenticated share access read/write access.
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: Prioritize Remediation
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="Remediation"/>
|
|
<label class="tabLabel" onClick="updateTab('Remediation',false)" for="Remediation"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>Recommendations: Prioritize Remediation</h2>
|
|
Below are some tips for getting started on prioritizing the remediation of shares configured with excessive privileges.
|
|
</div>
|
|
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px; margin-bottom:10px;"></div>
|
|
|
|
<table class="table table-striped table-hover tabledrop">
|
|
<thead>
|
|
<tr>
|
|
<th align="left">Share Access</th>
|
|
<th align="left">Impact</th>
|
|
<th align="left">Description</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr>
|
|
<td>High Risk Shares</td>
|
|
<td>Confidentiality, Integrity, Availability, Code Execution<br>
|
|
High likelihood.
|
|
</td>
|
|
<td>Remediate high risk shares. In the context of this report, high risk shares have been defined as shares that provide unauthorized remote access to systems or applications. By default, that includes wwwroot, inetpub, c$, and admin$ shares. However, additional exposures may exist that are not called out beyond that.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Write Access Shares</td>
|
|
<td>Confidentiality, Integrity, Availability, Code Execution</td>
|
|
<td>Remediate shares with write access. Write access to shares may allow an attacker to modify data, insert their own users into configuration files to access applications, or leverage write access to execute code on remote systems. Folders that provide write access could also fall victem to ransomware attacks.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Read Access Shares</td>
|
|
<td>Confidentiality,Code Execution</td>
|
|
<td>Remediate shares with read access. Read access may provide an attacker with unauthorized access to sensitive data and stored secrets such as passwords and private keys that could be used to gain unauthorized access to systems, applications, and databases.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Top Share Names</td>
|
|
<td>NA</td>
|
|
<td>Sub prioritize remediation based on top groups of share names(most common share names). When a large number of systems are configured with the same share, they often represent weak configurations associated with applications and processes.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Top Share Groups</td>
|
|
<td>NA</td>
|
|
<td>Sub prioritize remediation based on top share groups that have the same list of files in their directory. This is another way to identify systems that are configured with the same share are associated with the same insecure application deployment or process.
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Sub Prioritzation Tips</td>
|
|
<td>NA</td>
|
|
<td>
|
|
Use the detailed .csv files to:<br><br>
|
|
1. Identify share owners with the ShareOwner field. Filter out "BUILTIN\Administrators", "NT AUTHORITY\SYSTEM", and "NT SERVICE\TrustedInstaller" to identify potential asset owners.<br><br>
|
|
2. Filter out shares with a FileCount of 0.<br><br>
|
|
3. Sort shares by LastModifiedDate.<br><br>
|
|
4. Filter for keywords in the FileList.<br><br>
|
|
For example, simple keywords like sql, database, backup, password, etc can help identify additional high risk exposures quickly. <br>
|
|
</td>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<!--
|
|
|||||||||| PAGE: Home
|
|
-->
|
|
|
|
<input class="tabInput" name="tabs" type="radio" id="home"/>
|
|
<label class="tabLabel" onClick="updateTab('home',false)" for="home"></label>
|
|
<div id="tabPanel" class="tabPanel">
|
|
<div style="min-height: 670px">
|
|
<div style="margin-left:10px;margin-top:3px">
|
|
<h2>SMB Share <span style="color:#CE112D;">Excessive Privilege Report</span></h2>
|
|
This report summarizes the shares identified as being configured with excessive privileges.
|
|
</div>
|
|
<div style="border-bottom: 1px solid #DEDFE1 ; background-color:#f0f3f5; height:5px"></div>
|
|
|
|
<!--
|
|
|||||||||| CARD: SCAN SUMMARY
|
|
-->
|
|
|
|
<a href="#" id="DashLink" onClick="radiobtn = document.getElementById('dashboard');radiobtn.checked = true;">
|
|
<div class="card" style="position:absolute;margin-top:20px;width:270px;">
|
|
<div class="cardtitle" align="left">
|
|
Scan Summary<br>
|
|
<img style="vertical-align:top;float: right;clear:both;" src="data:image/png;base64, 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IgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOjQ5MjcwMDI1LWE0MDItNDY2Zi04ZTc3LWIxMjE5MzhjMTg1ZiIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOmJjNmYyOGRlLWYxOTQtNDBhMS04YjlkLTRhOTBjMTMwMTc5NCIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMzAwNDg4OTM2NjQ4IgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6OWIzMWFhMDMtMTM5ZC00MTYwLWI2MzUtNDE5NTc2MjIxODFjIgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI3VDEwOjIxOjI4Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAg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" />
|
|
<span class="cardsubtitle2"><br></span>
|
|
</div>
|
|
<div class="cardcontainer" align="left">
|
|
<span align="left">The scan context and run time information have been provided below.</span>
|
|
<table>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Domain</td>
|
|
<td >
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$TargetDomain</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">DC</td>
|
|
<td >
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$DomainController</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align: top;">Start Time</td>
|
|
<td>
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$StartTime</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Stop Time</td>
|
|
<td >
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$EndTime</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Duration</td>
|
|
<td >
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$RunTime</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Src Host</td>
|
|
<td >
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$SourceHost</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Src IPs</td>
|
|
<td >
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$SourceIps</span>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="cardsubtitle" style="vertical-align:top">Src User</td>
|
|
<td >
|
|
<span class="AclEntryRight" style="width:160px;word-wrap: break-word;">$username</span>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
|
|
<!-- home text -->
|
|
<div style="margin-left:300px;">
|
|
<br>
|
|
<h4>Collection Approach</h4>
|
|
<div>
|
|
The <a style="color:#333" href="https://github.com/NetSPI/PowerHuntShares/blob/main/PowerHuntShares.psm1">PowerHuntShares</a> audit script was run against the netspi.local domain to collect SMB Share data, generate this HTML summary report, and generate the associated csv files that detail potentially excessive share configurations.
|
|
The left menu can be used to find summary data, the scan summary is in the table to the left, and a summary of the data collection approach has been outlined below.<br>
|
|
<br>
|
|
<div style="position:relative;height:120px;">
|
|
<!-- LDAP QUERY -->
|
|
<div class="Minicard">
|
|
<div class="Minicardtitle" align="center">
|
|
Find Domain Computers<br>
|
|
</div>
|
|
<div class="Minicardcontainer" align="center">
|
|
|
|
<button class="collapsible">
|
|
<img style ="padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;margin-bottom:5px;" src="data:image/png;base64, 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Oi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvbW0vIgogICAgeG1sbnM6c3RFdnQ9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZUV2ZW50IyIKICAgIHhtbG5zOmRjPSJodHRwOi8vcHVybC5vcmcvZGMvZWxlbWVudHMvMS4xLyIKICAgIHhtbG5zOkdJTVA9Imh0dHA6Ly93d3cuZ2ltcC5vcmcveG1wLyIKICAgIHhtbG5zOnRpZmY9Imh0dHA6Ly9ucy5hZG9iZS5jb20vdGlmZi8xLjAvIgogICAgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIgogICB4bXBNTTpEb2N1bWVudElEPSJnaW1wOmRvY2lkOmdpbXA6NjBhMGY0NjUtNjJmYS00ZmVlLTk5NGUtNmYwYzc5ZmIwODhkIgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOmM2NmZkYzhhLTgxMjMtNDBiYS1iYTNiLTg2YTIwNDE2MGIxMSIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOmE1MjRlOTA5LWU1MjEtNGZiMC05MmMyLTExMzZjOWU1NjE1ZCIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMjM3NzY2Mjc4NDAwIgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6OTFjYWZiZWEtOTFiZi00YzEwLTliMzMtM2I0MWYyZmQ4NTE3IgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI2VDE2OjU2OjA2Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/PiNxLrcAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWOAboWgp4AAADyklEQVRIx61WX2hbZRT/ne+7N1mWtUnadLrYdquxoHQES1qx0251DJniH8YUfRN8ENGXPU4QwRcffHIviogvCiKCA1FhwjbBB6FL7FilnYRRS6hNejfnTdLb/L3f8aE39SZL0lvwwIXLd757fuec3/l936VTp+YI/5n7vWncy18uV2AYBgUCAT54cIg7fA8NvY27OVZWVmU2u5awbfs0gBEARSHo11AodDmRmLA0bSc0kasS2gWAAICZsbBwo980Cx8w82sAfACU6/uUrutvzcw89rvPpwMAhOP0BAAAmcwt3TQLF5j5DQAZInpT0+RTQoizAC4CmK7X69/Mz6cPM2+HoUgk1G9Z5fsApu1YDMcHInLwCERQUkqrUqkmlVLfElE6ENh3Npl8dMPv94OZsbz8hy+fN95l5neI6PPx8fjbo6PDigD8DCDplNyVGyJaBHAFQJyZXxVCPHfixJNXpBQtVafTCyHTLF4DEAqHQw9PTU0WNACTRPSRlOIHZjBtp99pAFgIYdRqta8BVA8cCF5rBwBA0eigVSiU0sx8xrKsGABTc9qyMjc3m+oxqgwAi4tLPsO4XQcgbNv2ddpv24qYeZ+zXG8SjyZBu01ZNDrQIKLrAPzlcuVZy7Ja9jIzcrmN+wEcA/BXONy/vgPi6lDPKYvFDikpxZcAtpRS76dS16ez2TXJzDDNAs3Pp6PVauUCgCEi+ioWO1QGQBLAeSK6NDZ2eNGjEDdMs1BlxotKqVfu3v3nodXVbHx9Pf98rVb7EMATTvKiWCxdGh6ObbpBbnhRejgcZqVUulgs/cnMUwCOM/PTAGYA6ET0I4AHAIw3Go3pfN74iQCYQohzJ08e/2I3ADdvpVIJmcytvmJxM8msRgAqCEGmbasXiCillPoYQAjAZc3rWdXOWV9fH5LJyZKjMwDA1au/vM7MLw8MRN4zTbNq2+pTABGxB4CO2mlbEwAgpeBE4uh3fr9vav/+wGnNY0Daq39wcEDNzh7L7hz1zCyXlm7qO+kIAaUUhBCQUnYMopTi5l6XSbfPfZ8IZj6Xy228tGs51FlKLi2POKfBPZcWATjqPL1JYE/U5doTEQBs/L9m67rWgqIR0SfM/EwPcvcydUxE3weDwWpLm8fGRrVSaVPTdb0F5M6dvycaDftMcyzbeZdSXhwaGly2bQWlbAYAXdcpGAzWjxwZbemOFo8/aLtaRi5hPc7M57uBKKXWJiYeWfAy/lq3WReC0sz0WTcQIeg3r/pq/q3cw8fWVhmGcZu2SaQWeogIkUhYBQIB9iBQ/hfq8Z63CHuDFwAAAABJRU5ErkJggg==" />
|
|
|
|
</button>
|
|
<div class="content">
|
|
Get a list of domain computers by querying a domain controller via ldap.
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- PING SCAN -->
|
|
<div class="MinicardconnectLine"></div>
|
|
<div class="Minicard">
|
|
<div class="Minicardtitle" align="center">
|
|
Ping Scan<br>
|
|
Computers
|
|
</div>
|
|
<div class="Minicardcontainer" align="center">
|
|
|
|
<button class="collapsible">
|
|
<img style ="padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;margin-bottom:5px;" src="data:image/png;base64, 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Za97vLunc27/9rTm9wMtE3KLJ+uWiAAADRhpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+Cjx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IlhNUCBDb3JlIDQuNC4wLUV4aXYyIj4KIDxyZGY6UkRGIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyI+CiAgPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIKICAgIHhtbG5zOnhtcE1NPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvbW0vIgogICAgeG1sbnM6c3RFdnQ9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZUV2ZW50IyIKICAgIHhtbG5zOmRjPSJodHRwOi8vcHVybC5vcmcvZGMvZWxlbWVudHMvMS4xLyIKICAgIHhtbG5zOkdJTVA9Imh0dHA6Ly93d3cuZ2ltcC5vcmcveG1wLyIKICAgIHhtbG5zOnRpZmY9Imh0dHA6Ly9ucy5hZG9iZS5jb20vdGlmZi8xLjAvIgogICAgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIgogICB4bXBNTTpEb2N1bWVudElEPSJnaW1wOmRvY2lkOmdpbXA6NTJkNWZlN2EtMzk3Yy00NTBkLTgxOTUtZmVkZTJlMjEyNTczIgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOmQzMWQxNTNlLTEyNjItNGE0ZS05ZmEyLWEwMTI1NWNjYWZlNSIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOmE0MGFlYmM2LTU5ZTAtNDIwYy04YzEwLTk5NzAxNzA4YmFjMCIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMjM3Nzg0MzgzMjEzIgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6MTljNTNjOGUtNzFlNC00MDhlLThjNDMtYzgwM2M3MTc1NjM2IgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI2VDE2OjU2OjI0Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/Pu/LGbMAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWOBgSVTcbAAAHTElEQVRIx4WWXUiTjxfHv8+zzfmImlMznc53SXMTndGFIDKCMcILIw1F08w36CpCMC2ibpTCugokUSox0C6SjJBk4tA0JCfObWC+rtThS7ZNt+fRuXn+Fz+Sf/yK34Fzcy7Ol3MOfM6XEQSB8Jfwer1YWVmBxWKByWSC2WzG9PQ01Go1VCoVsrOzoVQqkZycDKlU+rc2YP4kcnR0hJmZGbx+/Rp6vR4bGxuQ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" />
|
|
|
|
</button>
|
|
<div class="content">
|
|
Perform ping scanning to determine basic connectivity to domain computers.
|
|
</div>
|
|
|
|
</div>
|
|
</div>
|
|
|
|
<!-- PORT SCAN -->
|
|
<div class="MinicardconnectLine"></div>
|
|
<div class="Minicard">
|
|
<div class="Minicardtitle" align="center">
|
|
Port Scan<br>
|
|
TCP 445
|
|
</div>
|
|
<div class="Minicardcontainer" align="center">
|
|
|
|
<button class="collapsible">
|
|
<img style ="padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;margin-bottom:5px;" src="data:image/png;base64, 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LnhtcAAAAAAAPD94cGFja2V0IGJlZ2luPSLvu78iIGlkPSJXNU0wTXBDZWhpSHpyZVN6TlRjemtjOWQiPz4KPHg6eG1wbWV0YSB4bWxuczp4PSJhZG9iZTpuczptZXRhLyIgeDp4bXB0az0iWE1QIENvcmUgNC40LjAtRXhpdjIiPgogPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4KICA8cmRmOkRlc2NyaXB0aW9uIHJkZjphYm91dD0iIgogICAgeG1sbnM6eG1wTU09Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9tbS8iCiAgICB4bWxuczpzdEV2dD0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL3NUeXBlL1Jlc291cmNlRXZlbnQjIgogICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIgogICAgeG1sbnM6R0lNUD0iaHR0cDovL3d3dy5naW1wLm9yZy94bXAvIgogICAgeG1sbnM6dGlmZj0iaHR0cDovL25zLmFkb2JlLmNvbS90aWZmLzEuMC8iCiAgICB4bWxuczp4bXA9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC8iCiAgIHhtcE1NOkRvY3VtZW50SUQ9ImdpbXA6ZG9jaWQ6Z2ltcDo2OTQ2MTY3ZC00YzhmLTQ4OGQtYTRjNC1lOWU5MWEyM2I0ZmUiCiAgIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6YzUwNjFhNDAtNThkYi00NzNlLWE2ZDItYzcxYWU1N2U4YTI3IgogICB4bXBNTTpPcmlnaW5hbERvY3VtZW50SUQ9InhtcC5kaWQ6NTk2NjIzMmUtOGRkYy00ZDg0LWFiZTAtYzYwNzkwNTkzYmY4IgogICBkYzpGb3JtYXQ9ImltYWdlL3BuZyIKICAgR0lNUDpBUEk9IjIuMCIKICAgR0lNUDpQbGF0Zm9ybT0iV2luZG93cyIKICAgR0lNUDpUaW1lU3RhbXA9IjE2NDMyMzYyMzk1MDUyODYiCiAgIEdJTVA6VmVyc2lvbj0iMi4xMC4yOCIKICAgdGlmZjpPcmllbnRhdGlvbj0iMSIKICAgeG1wOkNyZWF0b3JUb29sPSJHSU1QIDIuMTAiPgogICA8eG1wTU06SGlzdG9yeT4KICAgIDxyZGY6U2VxPgogICAgIDxyZGY6bGkKICAgICAgc3RFdnQ6YWN0aW9uPSJzYXZlZCIKICAgICAgc3RFdnQ6Y2hhbmdlZD0iLyIKICAgICAgc3RFdnQ6aW5zdGFuY2VJRD0ieG1wLmlpZDphNTRlYzVhZi0wZTI5LTQyYzktOGI2MC01ZDBlMmI0MTQzMjIiCiAgICAgIHN0RXZ0OnNvZnR3YXJlQWdlbnQ9IkdpbXAgMi4xMCAoV2luZG93cykiCiAgICAgIHN0RXZ0OndoZW49IjIwMjItMDEtMjZUMTY6MzA6MzkiLz4KICAgIDwvcmRmOlNlcT4KICAgPC94bXBNTTpIaXN0b3J5PgogIDwvcmRmOkRlc2NyaXB0aW9uPgogPC9yZGY6UkRGPgo8L3g6eG1wbWV0YT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgIAo8P3hwYWNrZXQgZW5kPSJ3Ij8+Ewqy6gAAAIpQTFRFAHyt////AAAA/Pz8+Pj48vLy7u7u9fX13t7e4eHh8PDwvLy8VlZW2dnZm5ub6+vr0tLSxMTEkpKSsrKyzMzMqampGxsbCAgIT09PFRUVIiIioaGhdXV1z8/Pbm5uQEBAKCgoXl5ef39/Ly8vZ2dnJycnjIyMOTk5XFxcPj4+lZWVSkpKeXl5hISE0VcdHQAAAAF0Uk5TAEDm2GYAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWHidn7ZkCAAABB0lEQVQoz22S63KEIAxGJVGRsop1F7xtdQGr07G+/+sVsZ2pl/w9k+TjhCAIyEUhCYJrQq4IxkxEVz3hR9E9ZYIngvlQAzQLOxFalODq834isfakNEcS8R48WXYEo3yca99juScYCfaGGKVZVTbzzdb1VyFWgqk0ps1zqaGcR/5ol+UufOrEKIBa6wb6hQtEKgTdHLTKL4XpmzPc2dEb6CULD94Gnwaqd3o0KrdpTSeTcE/YaJWyvVXT4Aa69C7B5hrZmpq3w6SagSePZb5lf65DIUISprKbQOlqXWv2rqljry2oOrqmaWvh2jVhv1cYz5fL1D/Xu2snpi5Lm8XnH+Le8DQ8JsEPMhITTys21gwAAAAASUVORK5CYII=" />
|
|
|
|
</button>
|
|
<div class="content">
|
|
Perform port scanning on tcp port 445 to determine targets for share enumeration.
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- SHARE SCAN -->
|
|
<div class="MinicardconnectLine"></div>
|
|
<div class="Minicard">
|
|
<div class="Minicardtitle" align="center">
|
|
Enumerate<br>
|
|
Shares
|
|
</div>
|
|
<div class="Minicardcontainer" align="center">
|
|
|
|
<button class="collapsible">
|
|
<img style ="padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;margin-bottom:5px;" src="data:image/png;base64, iVBORw0KGgoAAAANSUhEUgAAABkAAAAZCAYAAADE6YVjAAAOk3pUWHRSYXcgcHJvZmlsZSB0eXBlIGV4aWYAAHja7ZlZcuS4EYbfcQofAVsigeNgjfANfHx/CZY0LbV6ZjT95AirQsUSiwSBXP4Fcvs//z7uX/zklIvLorW0Ujw/ueUWOx+qf376fQ8+3/fnj/b6Lnw8796/iJxKHNPzZy2v69/Oh/cBnkPnk/wwUJ2vL8bHL1p+jV8/DRSfQ7IZ2ef1Gqi9Bkrx+SK8BujPsnxpVX9cwtjP8XX/EwZ+nb3l+nHaP/2tRG8Jz0kx7hSS5z2m+kwg2W90qfMh8B5Tifap3M/+OfMajIB8Faf3H+Lsjk01f3nRh6y8fwpfn3efs5Xj65L0Kcjl/fjleRfk0xfp/Tnxxyfn+voUP57v059nRp+ib7/nrHrumllFz4VQl9ei3pZyP3Hd4BH26OqYWvHKrzCE3lfjVanqSSksP/3gNUMLkXSdkMMKPZyw73GGyRRz3C4qH2KcMd2TNWlscd5MZnuFEzW1tFIli5O0J87G97mE+9jmp7tPqzx5BS6NgcGC1cV3X+67N5xjrRCCr++xYl4xWrCZhmXO3rmMjITzCqrcAL+9Pv9YXhMZFIuytUgjsOMZYkj4AwnSTXTiQuH49GDQ9RqAEPFoYTIhkQGyFpKEErzGqCEQyEqCOlOPKcdBBoJIXEwy5kQXaazRHs0tGu6lUSKnHecBMzIhdJmSm5Y6ycpZqB/NlRrqkiSLSBGVKk16SSUXKaVoMVDsmjQ7FS2qWrVpr6nmKrVUrbW22ltsCdCUVpq22lrrnWd2Ru7c3bmg9xFHGnmIG2XoqKONPimfmafMMnXW2WZfcaUFfqyydNXVVt9hU0o7b9ll66677X4otZPcyUdOOXrqaae/Z+2V1p9e38haeGUt3kzZhfqeNc6qvg0RDE7EckbCosuBjKulgIKOljNfQ87RMmc58y3SFRKZpFjOVrCMkcG8Q5QT3nLn4pNRy9xv5c1p/pC3+E8z5yx138zcz3n7KmvLaGjejD1daEH1ie7j+117rN3I7qej+9UX3z3+f6D/iYH6yunsXpFCIj26XkY9O5xZevjFLSP36kHH2teu6azcV9n2iUIE5a2Jz6ayPQwkQfYZx289O+tciePqdhE1qP2oyaVTI+OewXA//+X+7Mvv/OX+5Msy9hwQvKZD9+8qO7Tj+65DF1BJj1ZecZ049nFrrk2Y5g6z7gnZiLYBId9ltYaWSHSsFFRFAP5GVCJGS/PpTBtDddq1rtQ12hwBdi/tCFPJp1ZIZtPDyS4BY1blubOdRPiWnUsjFQ1Dw8rNpjvFiWbZsaEy7IIZ8zqFj6R2JF0NWDhzyDpjaOGrUPl0+lKog7yHLOE5uueDDsP96CmEO9Hu9eav3TnV3olNP/kuuMpQOwLk3NOeIEyCTYnozFJCGb3swcMAzxYATOQs8+h7w8HN7ot5a29J6ki7B5Wum7nGOba4fUjTAWHb1Lx7WhsQnFqJD2kRaRQRSdiWg6E7EyctyaYsiD5qU5gsGXaUeUgsXM7phaiSch2cFCCaaeveB56YxGeV6uM4VdOU0Te4Pgx/k2HwiK7ktXj0hDjq7M0r4gHVpXrX7kdOsxCoTJQtGQ1WOKS1z0yvbRJHaFMrCTrqPIxfKgX5d4g6qSXLo1KBY1SQ/hTtxFBXHyZTbWqpxzagAa0bVmpBEBETCQbL8tRc09KzZrcKqFk3k2OymdXn0cdaJW56MKFXqcLAOplgq9m30Jy3wPlNfE/qB5LKlVrttPAIAyKbJY5Rji9nCiVSrVLSHJ3UQ610wNwDBZvcLizszD4LTKplG9vSXSSRlqfjOlxKFT31VJrcKmtPRcUuxl+d8FV3v3u+uecxUN3gx0ri7wzwdr/73QHe7ne/O8Db/e41gFXk6jGRE5QKN2QUAWVC+WxsxO2nlTpxX7gHLsNQ3DavuZyyfXGgVtnvFxJdjF8ck36h8MfzRCX1TajEEk7jvbedyK1lKYFH6IQ2XZmbjqSz9xfQ/2Dm3yIWF8fylHo9tSGqWKM9kRrpIBhwdOfKDCnpNcbevJ10NIepRhh6MnSkIwP+Xae2HaCpMelsNQzp1rEJrEkMir/yIDFYuHPfLB/8th6ma9pFS8Pl4Ja1v8yzALgw7DkjMw2a7xCnB8iQeB13pWlNSK6AzTmWRJMTH2zBXaH7i6VvJpYWMAWp4hpiRt15NGDFOPS8j4S5fdhb3TJaoeHpLkLE74BuZq/0Nv7R2ggkUORt2gCQpIvFswKkkKwcIP50H0Ucwv0meVYeyEnakuLzQpEB4b7ZRDqcFM0aFpkWtqvmLUCNh24wEh5xhHptq7gG5U0K2wgIhCgMFzplWmz34rKF/HFUTNPHUw4dTvwrgrUnqKzMuYhCSqHaHhDonFuqZEMWIpoIm1MesFKYmTaZJkYKlBFcWT4DHxWVn5AjJ5pF7jXAwCykw5VMMJCx3URV1ugaYJKJJ+0DEWOXDZno7IpNMLIbQyBs6ELAsH4ReislzwlwexZgGyMBzGeK0K7wkaemQEkdKsrZDWhzXH6iWedJt3ZIGQUCSuNNWHQjgrT60GwVnlkRcaP48B92dds/DEQd06M2B5rxxEhxsww5e8OCGWb4ehQbA70yqwN84yMWnnPPsMyhLBp9PJMBNpAhRyXBNHTAh8UTDbrHSQZ90oQ94uv29NwugRTePyALE4r9eSLPsxMpTw2YkR0tinLcXi1qt78QKCANgPQK5SQ6jFPttmifA5x7n3drrdF6zZbMintcQG1eqKPXg5M+4cN88RyjxEBdFhruWNpYFYUO0mhDHrW6TZUqFdKWm/hCvGiFozCZIG7CM0mi/WA+Us3pFE2jwWi4too+QOwImLPv0HVFegsNaVuGtDTPUDFJABKTkgmqrDCq5hbCNGfnU7Oth+Vh4z1MPQUIkOkZGNOGrl8pR7dhUM+OHftn20kNZUm0L5AgLBB+zObmlhzKjRIxNC6ZFGwLIOQaE1xPiL+0TdxUs/F2YRBmRnoRubEJlLKgXhOZexkzhIaIQilm3pgaamT4ibltSJmFt17+VrbIiqZYJOwYHjFopXkRMyhfVUIFVhp0Sgc53aT8MM64awwoiFzSMvC/wpS0XaxK8xUA1Medazf90i+K8giOAJugfj3CSRrZvDuduZlG7pviiFsCfGWbYCVnno7WmxiPOe8DcLlx3G4Yx4GVgH8nWwCJAMoaU8ePM02MfarRVxjP4LsPmKaC2tP2Kole8Y8naOj5gzzG948KF5CFsrhp60AmtY8y+i+PLvz9G5gu0CUzYxRigKeR02hCgAAB68ytUQun3bVS4shyD+YFGId10M+IrWBOZjbgYJtTMZwxHWnbWa+ADyeMatBbHiExgNKzaY5OQJCSdBBkuVrevjaBsOY2DUTRoBnkgR9Pqak7KGE1AqIwIDhw/NLMnV/Fb5moRelDORNGAFeitXhbtlf3I484+Zlr/tHxbSBBCmBrzQShGJKFtoz8ILKfJhOmv7WTQ5LVlsZjO3wErw+uGNOdlwMDGfF3KLFN4oG+98dJBK/WUPHtZAtLX2BeGZg07EU0CFuzqGveVpw32Auyx43XaQBMoLVKAkcAOXyzKRVI90YORyF9o6Tz3oByhd8gOAevouwR9efROiQiLKpVUgJMeqZ8o6du0jFhWH5Zau4bxfvlUdZK2zaiINFtPkqtXtA8OKcQUQMUaoIEzDuxCiUCj5fBniU1JWq+qKQSzJx1gs31aNhDSTVF/OJj0AQNadKAuf10ue9ClfbizXqmRwE3BOTC80V8S4xYiKY4NSRfBcK0EL/wIBAWa7dPyhbJQg1s2chFcmThRZ8hnkDGS9ndxCUQj3y8aAz0YRTJzaPLwl4UFBm8RFm+jpf7nUD/NBDyYIenim1boXoiMpf4Zz8Cj5tNTHjj5m3RriwNzYUan88WQzfFBlOD+5j8Tt1QcZDbbQWaxjY4UI22nWGyHTbkOm+mxjd6w7IIKAAxQI9bPlKMOHVORBN7BvEGKwEIv8lb9AJiDT17JCqZeeMSgxsbLBHi6LrHo5bTkF8JuQ+oQIjdtNtYYtVkN9Ec5nWtsIZhNoTU0CTFdm0qvoJ0Yvzs6Xhf1PzxAwoEcihE08sklibDo3wDRv4JDsmu2+RQXZJtj80ymCwRmBUCbdviAAqFF6hYFKHYRtPE71O/VCmtRE1jmDpXz7B6XUt6dmbiId2akWiyckqpqM9F0Wxl4OzMadBauMHXthBGsTWkzEnK8EMnOhckNLsuhGPbph26u6yOCV3G5pD5IabkzHaQKw0Dz47zMNO56nwRUuiy9SgOpTbNK3iowgvJPIgYBU8xHbc9xGJuwpygDASRMTIJQ5YioRX9PfGwWFFwkVZjQiry9Jkcc0l97m/tcLrvbYnyyf73WtDD1ZsqzvnZc5umsx+lvwItNm1vC+qFfmG5SwONCJSa+jSVgpJDZgB3J0VSxMUCVKxpGrLXjdUV7aw25WWbPNN2sxDpSFclPZigvsbWnHG5YM3BhNZt/zigy6bZvRiTM+Ch18O4EsoUcwEnX/op3rZs4BIyeEb7fxt2aZ5s9EupSMLWhhF83Ci2jUytJFQrqRkmtPaw/ztIbxEpNxfrxUPRe1l/jUrut+DM/hdn7RhXc3WuPVkTAgZgoPul7R/9CQgGxJhDhnJjskZf9q9xhf3itO0HtD1pcgDOxZfwkjXUHZ0E+GA/Di6zZvwEQTT6MTeBJzf6BD4IdDf3XKF5xUHGWcx9AWumc+XDzshtp7vDhcHPuhqpxXxs2ywvdglCvDZYBA0JN56MUzNJjlvZ0qwjFkLre8Di/taFSLQAchYR07y2tUphTbPSWKzaGytHseXHN3YUNJPVXW2/Dq0LgRJPA1MgfUTqN9c6BsK+xW1BXchfcwfUWiwPQfpy94oftrC9YhyOsQVAojvs20R4smF4TxjPAKMIJX+Ug1AoVTvdPzISNErtqydyi4phEtS0/ZsOAtif9qYONfNfEFTtvoAOxpEAAAGFaUNDUElDQyBwcm9maWxlAAB4nH2RPUjDUBSFT1OlohUHi0hxyFCdLIiKOEoVi2ChtBVadTB56R80aUhSXBwF14KDP4tVBxdnXR1cBUHwB8TRyUnRRUq8Lym0iPHC432cd8/hvfsAoVFhqtk1AaiaZaTiMTGbWxUDr/AhjCH0wS8xU0+kFzPwrK976qa6i/Is774/q1/JmwzwicRzTDcs4g3imU1L57xPHGIlSSE+Jx436ILEj1yXXX7jXHRY4JkhI5OaJw4Ri8UOljuYlQyVeJo4oqga5QtZlxXOW5zVSo217slfGMxrK2mu0xpBHEtIIAkRMmooowILUdo1Ukyk6Dzm4Q87/iS5ZHKVwcixgCpUSI4f/A9+z9YsTE26ScEY0P1i2x+jQGAXaNZt+/vYtpsngP8ZuNLa/moDmP0kvd7WIkfAwDZwcd3W5D3gcgcYftIlQ3IkPy2hUADez+ibcsDgLdC75s6tdY7TByBDs1q+AQ4OgbEiZa97vLunc27/9rTm9wMtE3KLJ+uWiAAADRhpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+Cjx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IlhNUCBDb3JlIDQuNC4wLUV4aXYyIj4KIDxyZGY6UkRGIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyI+CiAgPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIKICAgIHhtbG5zOnhtcE1NPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvbW0vIgogICAgeG1sbnM6c3RFdnQ9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZUV2ZW50IyIKICAgIHhtbG5zOmRjPSJodHRwOi8vcHVybC5vcmcvZGMvZWxlbWVudHMvMS4xLyIKICAgIHhtbG5zOkdJTVA9Imh0dHA6Ly93d3cuZ2ltcC5vcmcveG1wLyIKICAgIHhtbG5zOnRpZmY9Imh0dHA6Ly9ucy5hZG9iZS5jb20vdGlmZi8xLjAvIgogICAgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIgogICB4bXBNTTpEb2N1bWVudElEPSJnaW1wOmRvY2lkOmdpbXA6YzczMGM1Y2ItYjRjMi00YmRhLThlMDAtMWI3MDI1NjMwMDBiIgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOmEzNGQyZDc5LTMxZjMtNDYwYy1hM2M2LTFiNjFiNjljZTMzMyIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOjNiZmI0Y2M5LWE0YWQtNDNjMy04ZTkwLTE1MDljNzkxNmU2NCIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMjM1NjI3NzQ0MTc0IgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6NjNjM2VmYTAtNmI3Yi00NGZhLWFlZjctY2EzMjJlYzA2OGRjIgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI2VDE2OjIwOjI3Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/PrSHjn8AAAAGYktHRADoAPQASsUjROMAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWFBuybQ0PAAABi0lEQVRIx+2Tv0vDQBTH3zVpKsUghCb2x2C4lBq4paClSwfBzCVDQVBxDO3m6tBJl+zdBHERlwz9E1ycrHSyOkibDo6FDqaLtudiixcVpd4i9MF3eXfw4XPvHsCiFvXvy5Fl+R4AWuEoitLBGDcqlcrqnwjJZLLjuu7Sd+e5XG5fkqTbUql0nMlk4vNyWj9d8DxPIIQcmqb5+JXxNNls9sayrO25IL+ter2updPpu3AfvUM2pw3Lspbb7fZJEARxWZZRNBoFSikAAFBKmYzHY5hMJrNQSmE0GtmSJDUFQYBYLAbFYvHhk0mhUDgCAMozkbDaYDDY4/l1CSFXDMS2bdLtdglPSCqVOmcgvV5vlydAUZRnURQ9Zia6rvs8Z4ExPgMAmJnk8/kN3/fXeJqYpnnJ7IlhGC5PC8MwnqaAmQlC6ICnRb/fP2U2vlwub/HejVqtts5AMMYNngBd168/AiIIIRgOhzs8n0pV1YtwL+BpIYpi4DjOCkPQNK2qadoLD4Cqqq+JRKIatngD35X+UDOXLeMAAAAASUVORK5CYII=" />
|
|
</button>
|
|
<div class="content">
|
|
Scan for a list of shares on accessible domain computers.<br><br>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- SHARE ACL SCAN -->
|
|
<div class="MinicardconnectLine"></div>
|
|
<div class="Minicard">
|
|
<div class="Minicardtitle" align="center">
|
|
Enumerate Share ACLs<br>
|
|
</div>
|
|
<div class="Minicardcontainer" align="center">
|
|
|
|
<button class="collapsible">
|
|
<img style ="padding-bottom:5px;border-bottom:1px solid #ccc; padding-left:10px; padding-right:10px;margin-bottom:5px;" src="data:image/png;base64, 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MC00M2FhLTk3MmEtMzM2MDEyNzQzY2RjIgogICB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOjU4MTg5M2UwLWFmZTktNDhkNi1hN2VlLThhMTU3MjA2MGQzMiIKICAgeG1wTU06T3JpZ2luYWxEb2N1bWVudElEPSJ4bXAuZGlkOjBlMjA5NmU2LTNmYWYtNGEzOS1hMjQyLWMxNDFmMWU4MDQzYiIKICAgZGM6Rm9ybWF0PSJpbWFnZS9wbmciCiAgIEdJTVA6QVBJPSIyLjAiCiAgIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiCiAgIEdJTVA6VGltZVN0YW1wPSIxNjQzMjM2OTc2MzY1NjM0IgogICBHSU1QOlZlcnNpb249IjIuMTAuMjgiCiAgIHRpZmY6T3JpZW50YXRpb249IjEiCiAgIHhtcDpDcmVhdG9yVG9vbD0iR0lNUCAyLjEwIj4KICAgPHhtcE1NOkhpc3Rvcnk+CiAgICA8cmRmOlNlcT4KICAgICA8cmRmOmxpCiAgICAgIHN0RXZ0OmFjdGlvbj0ic2F2ZWQiCiAgICAgIHN0RXZ0OmNoYW5nZWQ9Ii8iCiAgICAgIHN0RXZ0Omluc3RhbmNlSUQ9InhtcC5paWQ6MDYwMjZhYTEtZWQyYy00NDU2LWExZTAtYzdlZWVjZThiNDY1IgogICAgICBzdEV2dDpzb2Z0d2FyZUFnZW50PSJHaW1wIDIuMTAgKFdpbmRvd3MpIgogICAgICBzdEV2dDp3aGVuPSIyMDIyLTAxLTI2VDE2OjQyOjU2Ii8+CiAgICA8L3JkZjpTZXE+CiAgIDwveG1wTU06SGlzdG9yeT4KICA8L3JkZjpEZXNjcmlwdGlvbj4KIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAKPD94cGFja2V0IGVuZD0idyI/PvHyhAIAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfmARoWKjhRz2cAAAAA8klEQVRIx+3WMUoDQRTG8d9m13SpLMQLWNiqbTB4CAs7EcFKEI8hKdIpKfUWQhCxUtRK9AAWsbMRBMnabMBiSLLrrI15MDB8fDP/ebw3zDCPEpHgAEfIp3jfsI3XKqAXNGfwbeIRy1UgTyW8HdxjqQwgK3mgAY5xjY8JvrzwHCJPikxWsYiViLU+wSnOx5A2rnATubFaSMc12UKvpg5+bhSTFKOaIKPGX1zGOeSfQrJfrt/AekC/w20syFcxQnq0TNawF7qAeIgFucBlQB/GzGQH+wH9DP1YkP7PzSa1cIaFGd74KtFEmqCLXXzivYaPSvcbYO4pyzEBUV8AAAAASUVORK5CYII=" />
|
|
|
|
</button>
|
|
<div class="content">
|
|
Scan enumerated shares for ACL entries configured with excessive privileges.
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div style="float:left;display:block;position:relative;">
|
|
<h4>How do I use this report?</h4>
|
|
Follow the guidance below to get the most out of this report.
|
|
<br><br>
|
|
<button class="collapsible"><span style="color:#CE112D;">1</span> | Review Reports and Insights</button>
|
|
<div class="content">
|
|
<div class="landingtext" >
|
|
Review the reports and data insights to get a quick feel for the level of SMB share exposure in your environment.
|
|
<br><br>
|
|
<strong style="color:#333">Reports</strong><br>
|
|
The <em>Scan, Computer, Share, and ACL</em> summary sections will provide a summary of the results.
|
|
<br>
|
|
<br>
|
|
<strong style="color:#333">Data Insights</strong><br>
|
|
The <em>Data Insights</em> sections are intented to highlight natural data groupings that can help centralize and expedite remediation on scale in Active Directory environments.
|
|
<br>
|
|
</div>
|
|
</div>
|
|
<button class="collapsible"><span style="color:#CE112D;">2</span> | Review Detailed CSV Files</button>
|
|
<div class="content">
|
|
<div class="landingtext">
|
|
Review potentially excessive share ACL entry details in the associated HTML and CSV files.
|
|
</div>
|
|
</div>
|
|
|
|
<button class="collapsible"><span style="color:#CE112D;">3</span> | Review Definitions</button>
|
|
<div class="content">
|
|
<div class="landingtext">
|
|
Review the definitions below to ensure you understand what was targeted and how privileges have been qualified as excessive.
|
|
<br><br>
|
|
<strong style="color:#333">Excessive Privileges</strong><br>
|
|
In the context of this report, excessive read and write share permissions have been defined as any network share ACL containing an explicit entry for the <em>"Everyone", "Authenticated Users", "BUILTIN\Users", "Domain Users", or "Domain Computers"</em> groups.
|
|
All provide domain users access to the affected shares due to privilege inheritance.
|
|
<Br><br>
|
|
Please note that share permissions can be overruled by NTFS permissions. Also, be aware that testing excluded share names containing the following keywords: <em>"print$", "prnproc$", "printer", "netlogon",and "sysvol"</em>.
|
|
<br><br>
|
|
<strong style="color:#333">High Risk Shares</strong>
|
|
<br>
|
|
In the context of this report, high risk shares have been defined as shares that provide unauthorized remote access to a system or application. By default, that includes <em>wwwroot, inetpub, c$, and admin$</em> shares. However, additional exposures may exist that are not called out beyond that.
|
|
<br>
|
|
</div>
|
|
</div>
|
|
|
|
<button class="collapsible"><span style="color:#CE112D;">4</span> | Verify and Remediate Issues</button>
|
|
<div class="content">
|
|
<div class="landingtext" >
|
|
Follow the guidance in the Exploit Share Access, Detect Share Access, and Prioritize Remediation sections.</div>
|
|
</div>
|
|
|
|
<button class="collapsible"><span style="color:#CE112D;">5</span> | Run Scan Again</button>
|
|
<div class="content">
|
|
<div class="landingtext" style="">
|
|
Collect SMB Share data and generate this HTML report by running <a href="https://github.com/NetSPI/PowerShell/blob/master/Invoke-HuntSMBShares.ps1">Invoke-HuntSMBShares.ps1</a> audit script.<br>
|
|
The command examples below can be used to identify potentially malicious share permissions.
|
|
<br><br>
|
|
<strong style="color:#333">From Domain System</strong>
|
|
<div style="border-radius:10px;border: 2px solid #CCC;margin-top:5px;padding: 5px;padding-left: 15px;width:95%;background-color:white;color:#757575;font-family:Lucida, Grande, sans-serif;">
|
|
Invoke-HuntSMBShares -Threads 20 -RunSpaceTimeOut 10 -OutputDirectory c:\folder\
|
|
</div>
|
|
<br>
|
|
<strong style="color:#333">From Non-Domain System</strong>
|
|
<div style="border-radius:10px;border: 2px solid #CCC;margin-top:5px;padding: 5px;padding-left: 15px;width:95%;background-color:white;color:#757575;font-family:Lucida, Grande, sans-serif;">
|
|
runas /netonly /user:domain\user PowerShell.exe<Br>
|
|
Import-Module Invoke-HuntSMBShares.ps1<br>
|
|
Invoke-HuntSMBShares -Threads 20 -RunSpaceTimeOut 10 -OutputDirectory c:\folder\ -DomainController 10.1.1.1 -Username domain\user -Password password
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<Br>
|
|
<br>
|
|
</div>
|
|
<script>
|
|
var coll = document.getElementsByClassName("collapsible");
|
|
var i;
|
|
|
|
for (i = 0; i < coll.length; i++) {
|
|
coll[i].addEventListener("click", function() {
|
|
this.classList.toggle("active");
|
|
var content = this.nextElementSibling;
|
|
if (content.style.maxHeight){
|
|
content.style.maxHeight = null;
|
|
} else {
|
|
content.style.Height = content.scrollHeight + "px";
|
|
content.style.maxHeight = "100%";
|
|
}
|
|
});
|
|
}
|
|
</script>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<!--
|
|
|||||||||| FOOTER
|
|
-->
|
|
|
|
<div style="float:right;border-top: 1px solid #DEDFE1 ; background-color:#f0f3f5;width:100%">
|
|
<img style="float:right;margin-right:15px;margin-top:15px;margin-bottom:15px;margin-right:50px;" src=""/>
|
|
<br>
|
|
</div>
|
|
</body>
|
|
</html>
|
|
"@
|
|
$NewHtmlReport | Out-File "$OutputDirectoryBase\Summary-Report.html"
|
|
# Write-Output " [*] Saving results to $OutputDirectory\_Report-$TargetDomain-Share-Inventory-Summary.html"
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Generate Excessive Privilege Findings
|
|
# ----------------------------------------------------------------------
|
|
if($ExportFindings){
|
|
|
|
Write-Output " [*] Generating exccessive privileges export."
|
|
|
|
# Define excessive priv fields
|
|
$ExcessivePrivID = "M:2989294"
|
|
$ExcessivePrivName = "Excessive Privileges - Network Shares"
|
|
$ExcessivePrivFinding = "At least one network share has been configured with excessive privileges. Excessive privileges include shares that are configured to provide the Everyone, BUILTIN\Users, Authenticated Users, or Domain Users groups with access that is not required."
|
|
$ExcessivePrivRecommmend = "Practice the principle of least privileges and only allow users with a defined business need to access the affected shares."
|
|
|
|
# Create a finding for each instance
|
|
$PrivExport = $ExcessiveSharePrivs |
|
|
Foreach {
|
|
|
|
# Grab default fields
|
|
$ComputerName = $_.ComputerName
|
|
$IpAddress = $_.IpAddress
|
|
$ShareName = $_.ShareName
|
|
$SharePath = $_.SharePath
|
|
$ShareDescription = $_.ShareDescription
|
|
$ShareOwner = $_.ShareOwner
|
|
$ShareACcess = $_.ShareACcess
|
|
$FileSystemRights = $_.FileSystemRights
|
|
$AccessControlType = $_.AccessControlType
|
|
$IdentityReference = $_.IdentityReference
|
|
$IdentitySID = $_.IdentitySID
|
|
$AccessControlType = $_.AccessControlType
|
|
$LastModifiedDate = $_.LastModifiedDate
|
|
$FileCount = $_.FileCount
|
|
$FileList = $_.FileList
|
|
|
|
# Create new finding object
|
|
$object = New-Object psobject
|
|
$object | add-member noteproperty MasterFindingSourceIdentifier $ExcessivePrivID
|
|
$object | add-member noteproperty InstanceName "Excessive Share ACL"
|
|
$object | add-member noteproperty AssetName $ComputerName
|
|
$object | add-member noteproperty IssueFirstFoundDate $EndTime
|
|
$object | add-member noteproperty VerificationCaption01 "$IdentityReference has $FileSystemRights privileges on $SharePath."
|
|
$ShareDetails = @"
|
|
Computer Name: $ComputerName
|
|
IP Address: $IpAddress
|
|
Share Name: $ShareName
|
|
Share Path: $SharePath
|
|
Share Description: $ShareDescription
|
|
Share Owner $ShareOwner
|
|
Share Accses: $ShareACcess
|
|
File System Rights: $FileSystemRights
|
|
Access Control Type: $AccessControlType
|
|
Identify Reference: $IdentityReference
|
|
Identity SID: $IdentitySID
|
|
Access Control Type: $AccessControlType
|
|
Last Modification Date: $LastModifiedDate
|
|
File Count: $FileCount
|
|
File List Sample:
|
|
$FileList
|
|
"@
|
|
$object | add-member noteproperty VerificationText01 $ShareDetails
|
|
$object | add-member noteproperty VerificationCaption02 "caption 2"
|
|
$object | add-member noteproperty VerificationText02 "text 2"
|
|
$object | add-member noteproperty VerificationCaption03 "caption 3"
|
|
$object | add-member noteproperty VerificationText03 "text 3"
|
|
$object | add-member noteproperty VerificationCaption04 "caption 4"
|
|
$object | add-member noteproperty VerificationText04 "text 4"
|
|
$object
|
|
}
|
|
|
|
# Write export file
|
|
$PrivExport | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Excessive-Privileges-EXPORT.csv" -Append
|
|
|
|
# Create record containing verification summary for domain
|
|
$object = New-Object psobject
|
|
$object | add-member noteproperty MasterFindingSourceIdentifier $ExcessivePrivID
|
|
$object | add-member noteproperty InstanceName "Domain ACL Summary"
|
|
$object | add-member noteproperty AssetName $TargetDomain
|
|
$object | add-member noteproperty IssueFirstFoundDate $EndTime
|
|
$object | add-member noteproperty VerificationCaption01 "$ExcessiveSharesCount shares across $ComputerWithExcessive systems are configured with $ExcessiveSharePrivsCount potentially excessive ACLs."
|
|
$ShareDetails = $ExcessiveSharePrivs | Select-Object SharePath -Unique -ExpandProperty SharePath | Out-String
|
|
$object | add-member noteproperty VerificationText01 $ShareDetails
|
|
$object | add-member noteproperty VerificationCaption02 "$TargetDomain SMB Share Scan Summary"
|
|
$Summary1 = @"
|
|
Target Domain: $TargetDomain
|
|
|
|
Scan Summary
|
|
Start Time: $StartTime
|
|
End Time: $EndTime
|
|
Run Time: $RunTime
|
|
|
|
Computer Summary
|
|
$ComputerCount domain computers found
|
|
$ComputerPingableCount domain computers responded to ping
|
|
$Computers445OpenCount domain computers had TCP port 445 accessible
|
|
|
|
Share Summary
|
|
$AllSMBSharesCount shares were found.
|
|
$ExcessiveSharesCount shares across $ComputerWithExcessive systems are configured with $ExcessiveSharePrivsCount potentially excessive ACLs.
|
|
$SharesWithWriteCount shares across $ComputerWithWriteCount systems can be written to.</li>
|
|
$SharesHighRiskCount shares across $ComputerwithHighRisk systems are considered high risk.
|
|
$Top5ShareCountTotal of $AllAccessibleSharesCount ($DupPercent) shares are associated with the top 5 share names.
|
|
|
|
The 5 most common share names are:
|
|
|
|
"@
|
|
|
|
$Summary2 = $CommonShareNamesTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$ShareName = $_.name
|
|
Write-Output "- $ShareCount $ShareName"
|
|
} | Out-String
|
|
|
|
$SummaryFinal = $Summary1 + $Summary2
|
|
|
|
$object | add-member noteproperty VerificationText02 "$SummaryFinal"
|
|
$object | add-member noteproperty VerificationCaption03 "caption 3"
|
|
$object | add-member noteproperty VerificationText03 "text 3"
|
|
$object | add-member noteproperty VerificationCaption04 "caption 4"
|
|
$object | add-member noteproperty VerificationText04 "text 4"
|
|
|
|
# Write record to file
|
|
$object | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Excessive-Privileges-EXPORT.csv" -Append
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Generate Excessive Privilege Findings - High Risk
|
|
# ----------------------------------------------------------------------
|
|
if($ExportFindings){
|
|
|
|
Write-Output " [*] Generating HIGH RISK exccessive privileges export."
|
|
|
|
# Define excessive priv fields
|
|
$ExcessivehighRiskID = "MAN:M:e581ab69-a0fc-4cb1-a7ff-87256c1a9e91"
|
|
$ExcessivehighRiskName = "Excessive Privileges - Network Shares - High Risk"
|
|
$ExcessiveHighRiskFinding = "At least one network share has been configured with high risk excessive privileges. High risk excessive privileges provide the Everyone, BUILTIN\Users, Authenticated Users, or Domain Users groups with read/write access to system shares, web roots, or directories containing potentially sensitive data."
|
|
$ExcessiveHighRiskRecommmend = "Practice the principle of least privileges and only allow users with a defined business need to access the affected shares."
|
|
|
|
|
|
# Create a finding for each instance
|
|
$PrivHighExport = $SharesHighRisk |
|
|
Foreach {
|
|
|
|
# Grab default fields
|
|
$ComputerName = $_.ComputerName
|
|
$IpAddress = $_.IpAddress
|
|
$ShareName = $_.ShareName
|
|
$SharePath = $_.SharePath
|
|
$ShareDescription = $_.ShareDescription
|
|
$ShareOwner = $_.ShareOwner
|
|
$ShareACcess = $_.ShareACcess
|
|
$FileSystemRights = $_.FileSystemRights
|
|
$AccessControlType = $_.AccessControlType
|
|
$IdentityReference = $_.IdentityReference
|
|
$IdentitySID = $_.IdentitySID
|
|
$AccessControlType = $_.AccessControlType
|
|
$LastModifiedDate = $_.LastModifiedDate
|
|
$FileCount = $_.FileCount
|
|
$FileList = $_.FileList
|
|
|
|
# Create new finding object
|
|
$object = New-Object psobject
|
|
$object | add-member noteproperty MasterFindingSourceIdentifier $ExcessivehighRiskID
|
|
$object | add-member noteproperty InstanceName "Excessive Share ACL"
|
|
$object | add-member noteproperty AssetName $ComputerName
|
|
$object | add-member noteproperty IssueFirstFoundDate $EndTime
|
|
$object | add-member noteproperty VerificationCaption01 "$IdentityReference has $FileSystemRights privileges on $SharePath."
|
|
$ShareDetails = @"
|
|
Computer Name: $ComputerName
|
|
IP Address: $IpAddress
|
|
Share Name: $ShareName
|
|
Share Path: $SharePath
|
|
Share Description: $ShareDescription
|
|
Share Owner $ShareOwner
|
|
Share Accses: $ShareACcess
|
|
File System Rights: $FileSystemRights
|
|
Access Control Type: $AccessControlType
|
|
Identify Reference: $IdentityReference
|
|
Identity SID: $IdentitySID
|
|
Access Control Type: $AccessControlType
|
|
Last Modification Date: $LastModifiedDate
|
|
File Count: $FileCount
|
|
File List Sample:
|
|
$FileList
|
|
"@
|
|
$object | add-member noteproperty VerificationText01 $ShareDetails
|
|
$object | add-member noteproperty VerificationCaption02 "caption 2"
|
|
$object | add-member noteproperty VerificationText02 "text 2"
|
|
$object | add-member noteproperty VerificationCaption03 "caption 3"
|
|
$object | add-member noteproperty VerificationText03 "text 3"
|
|
$object | add-member noteproperty VerificationCaption04 "caption 4"
|
|
$object | add-member noteproperty VerificationText04 "text 4"
|
|
$object
|
|
}
|
|
|
|
# Write export file
|
|
$PrivHighExport | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Excessive-Privileges-EXPORT.csv" -Append
|
|
|
|
|
|
# Create record containing verification summary for domain
|
|
$object = New-Object psobject
|
|
$object | add-member noteproperty MasterFindingSourceIdentifier $ExcessivehighRiskID
|
|
$object | add-member noteproperty InstanceName "Domain ACL Summary"
|
|
$object | add-member noteproperty AssetName $TargetDomain
|
|
$object | add-member noteproperty IssueFirstFoundDate $EndTime
|
|
$object | add-member noteproperty VerificationCaption01 "$SharesHighRiskCount shares across $ComputerwithHighRisk systems are considered high risk."
|
|
$ShareDetails = $SharesHighRisk | Select-Object SharePath -Unique -ExpandProperty SharePath | Out-String
|
|
$object | add-member noteproperty VerificationText01 $ShareDetails
|
|
$object | add-member noteproperty VerificationCaption02 "$TargetDomain SMB Share Scan Summary"
|
|
$Summary1 = @"
|
|
Target Domain: $TargetDomain
|
|
|
|
Scan Summary
|
|
Start Time: $StartTime
|
|
End Time: $EndTime
|
|
Run Time: $RunTime
|
|
|
|
Computer Summary
|
|
$ComputerCount domain computers found
|
|
$ComputerPingableCount domain computers responded to ping
|
|
$Computers445OpenCount domain computers had TCP port 445 accessible
|
|
|
|
Share Summary
|
|
$AllSMBSharesCount shares were found.
|
|
$ExcessiveSharesCount shares across $ComputerWithExcessive systems are configured with $ExcessiveSharePrivsCount potentially excessive ACLs.
|
|
$SharesWithWriteCount shares across $ComputerWithWriteCount systems can be written to.</li>
|
|
$SharesHighRiskCount shares across $ComputerwithHighRisk systems are considered high risk.
|
|
$Top5ShareCountTotal of $AllAccessibleSharesCount ($DupPercent) shares are associated with the top 5 share names.
|
|
|
|
The 5 most common share names are:
|
|
|
|
"@
|
|
|
|
$Summary2 = $CommonShareNamesTop5 |
|
|
foreach {
|
|
$ShareCount = $_.count
|
|
$ShareName = $_.name
|
|
Write-Output "- $ShareCount $ShareName"
|
|
} | Out-String
|
|
|
|
$SummaryFinal = $Summary1 + $Summary2
|
|
|
|
$object | add-member noteproperty VerificationText02 "$SummaryFinal"
|
|
$object | add-member noteproperty VerificationCaption03 "caption 3"
|
|
$object | add-member noteproperty VerificationText03 "text 3"
|
|
$object | add-member noteproperty VerificationCaption04 "caption 4"
|
|
$object | add-member noteproperty VerificationText04 "text 4"
|
|
|
|
# Write record to file
|
|
$object | Export-Csv -NoTypeInformation "$OutputDirectory\$TargetDomain-Excessive-Privileges-EXPORT.csv" -Append
|
|
}
|
|
|
|
#Write-Output " [*] Results exported to $OutputDirectory\$TargetDomain-Excessive-Privileges-EXPORT.csv"
|
|
}
|
|
}
|
|
|
|
|
|
# //////////////////////////////////////////////////////////////////////////
|
|
# Functions used by Get-SmbShareInventory
|
|
# //////////////////////////////////////////////////////////////////////////
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-CardCreationTime
|
|
# -------------------------------------------
|
|
function Get-CardCreationTime
|
|
{
|
|
[CmdletBinding()]
|
|
Param(
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Data table to parse. This should be the excessive privileges ACL table')]
|
|
$MyDataTable,
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Output file path.')]
|
|
[string]$OutFilePath = "Share-ACL-CreationDate-Monthly-Summary.csv"
|
|
)
|
|
|
|
# Get list of years for CreationdDate
|
|
$ExcessivePrivsAclCount = $MyDataTable | measure | select count -expandproperty count
|
|
$ExcessivePrivsYears = $MyDataTable | select CreationDateYear -unique | Sort-Object CreationDateYear
|
|
$ExcessivePrivsYearsCount = $ExcessivePrivsYears | measure | select count -expandproperty count
|
|
$ExcessivePrivsYearsfirst = $ExcessivePrivsYears | select CreationDateYear -first 1 | select CreationDateYear -ExpandProperty CreationDateYear
|
|
$ExcessivePrivsYearsLast = $ExcessivePrivsYears | select CreationDateYear -last 1 | select CreationDateYear -ExpandProperty CreationDateYear
|
|
|
|
# Create data table to store summary data for chart scale
|
|
$ChartDataSummary = New-Object System.Data.DataTable
|
|
$null = $ChartDataSummary.Columns.Add("Year")
|
|
$null = $ChartDataSummary.Columns.Add("Month")
|
|
$null = $ChartDataSummary.Columns.Add("Computer")
|
|
$null = $ChartDataSummary.Columns.Add("Share")
|
|
$null = $ChartDataSummary.Columns.Add("ShareAcl")
|
|
$null = $ChartDataSummary.Columns.Add("AclRead")
|
|
$null = $ChartDataSummary.Columns.Add("AclWrite")
|
|
$null = $ChartDataSummary.Columns.Add("AclHR")
|
|
|
|
# ///////////////////////// Get bar chart ranges
|
|
|
|
# Year Loop
|
|
$ExcessivePrivsYears |
|
|
foreach {
|
|
|
|
$TargetYear = $_.CreationDateYear
|
|
$TargetYearData = $MyDataTable | where CreationDateYear -like "$TargetYear"
|
|
|
|
# Month looop
|
|
1..12 |
|
|
foreach {
|
|
|
|
# do last day of month look up here
|
|
$currentMonth = $_
|
|
$monthnames = @(0,"January","February","March","April","May","June","July","August","September","October","November","December")
|
|
$monthdays = @(0,31,28,31,30,31,30,31,31,30,31,30,31)
|
|
$enddays = $monthdays[$_]
|
|
$currentmonthname = $monthnames[$_]
|
|
|
|
# setup start and end dates
|
|
[Datetime]$startDate = Get-Date -Year $TargetYear -Month $_ -Day 1
|
|
[Datetime]$endDate = Get-Date -Year $TargetYear -Month $_ -Day $enddays
|
|
|
|
# get data for the month
|
|
$MonthAcls = $TargetYearData | Where-Object {([Datetime]$_.CreationDate.trim() -ge $startDate -and [Datetime]$_.CreationDate.trim() -le $endDate)}
|
|
|
|
# Get acl count & and % of total
|
|
$MonthAclsCount = $MonthAcls | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclsCount -eq 0){
|
|
$MonthAclsCountP = 0;
|
|
}else{
|
|
$MonthAclsCountP = [math]::Round($MonthAclsCount/$ExcessivePrivsAclCount,4).tostring("P") -replace(" ","")
|
|
}
|
|
|
|
# Get share count
|
|
$MonthShareCount = $MonthAcls | select sharepath -Unique | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get computer count
|
|
$MonthComputerCount = $MonthAcls | select computername -Unique| Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get read count
|
|
$MonthAclReadCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*Read*") -or ($_.FileSystemRights -like "*Append*") } | Where-Object {($_.FileSystemRights -notlike "*GenericAll*") -and ($_.FileSystemRights -notlike "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get write count
|
|
$MonthAclWriteCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get hr count
|
|
$MonthAclHrCount = $MonthAcls | Where-Object {($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share')} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Populate table
|
|
$null = $ChartDataSummary.Rows.Add("$TargetYear","$currentmonthname",$MonthComputerCount,$MonthShareCount,$MonthAclsCount,$MonthAclReadCount,$MonthAclWriteCount,$MonthAclHrCount)
|
|
|
|
# Export Results
|
|
$ChartDataSummary | Export-Csv -NoTypeInformation $OutFilePath
|
|
|
|
} #End Month
|
|
} # End year
|
|
|
|
# Get highest ShareAcl
|
|
$HighestAclCountinMonth = $ChartDataSummary | select shareacl | sort {[int]$_.shareacl} -Descending | select shareacl -First 1 -ExpandProperty shareacl
|
|
|
|
# Get highest AclRead
|
|
$HighestAclReadCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclRead} -Descending | Select-Object AclRead -First 1 -ExpandProperty AclRead
|
|
|
|
# Get highest AclWrite
|
|
$HighestAclWriteCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclWrite} -Descending | Select-Object AclWrite -First 1 -ExpandProperty AclWrite
|
|
|
|
# Get highest AclHR
|
|
$HighestAclHRCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclHR} -Descending | Select-Object AclHR -First 1 -ExpandProperty AclHR
|
|
|
|
# Get highst ACL type count
|
|
$TypeCounts = $HighestAclReadCountinMonth,$HighestAclWriteCountinMont,$HighestAclHRCountinMonth
|
|
$HighestTypeCount = $TypeCounts | Sort-Object {[int]$_} -Descending | select -First 1
|
|
|
|
|
|
# ///////////////////////// Get bar chart data
|
|
|
|
# Start Table
|
|
$HTML1 = @"
|
|
<div class="LargeCard">
|
|
<div class="LargeCardTitle">
|
|
Share Creation Timeline<br>
|
|
<span class="LargeCardSubtitle2">for share ACLs configured with excessive privileges</span>
|
|
</div>
|
|
<div class="LargeCardContainer" align="center">
|
|
|
|
<div class="container" style="position: relative;float:left;bottom:0;left:0;height:195px;width:50px;">
|
|
<div style="top:5;position:absolute;color:#757575;width:100%;font-size:10;" align="right">$HighestAclCountinMonth <span style="color: #ccc">-</span> </div>
|
|
<div style="bottom: 98;position:absolute;color:#757575;border-bottom:1px solid #ccc;width:100%;font-size:10;padding-right:1px;" align="right">0 <span style="color: #ccc">-</span></div>
|
|
<div style="bottom:55;position:absolute;width:100%;font-size:10">
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">High Risk<br></div>
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">Write<br></div>
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">Read<br></div>
|
|
</div>
|
|
</div>
|
|
<div class="TimelineChart" Style="grid-template-columns: 1px repeat($ExcessivePrivsYearsCount, 204px) 1px;">
|
|
"@
|
|
|
|
$HTML1
|
|
|
|
# Year Loop
|
|
$ExcessivePrivsYears |
|
|
foreach {
|
|
|
|
$TargetYear = $_.CreationDateYear
|
|
$TargetYearData = $MyDataTable | where CreationDateYear -like "$TargetYear"
|
|
|
|
# Start Year
|
|
$HTMLYearStart = @'
|
|
<div class="YearItem" >
|
|
<div id="YearWrapper" style="float:left;background-color:#F2F3F4;">
|
|
<div id="MonthsWrapper" style="position: relative;float:left">
|
|
|
|
'@
|
|
$HTMLYearStart
|
|
|
|
# Month looop
|
|
1..12 |
|
|
foreach {
|
|
|
|
# do last day of month look up here
|
|
$currentMonth = $_
|
|
$monthnames = @(0,"January","February","March","April","May","June","July","August","September","October","November","December")
|
|
$monthdays = @(0,31,28,31,30,31,30,31,31,30,31,30,31)
|
|
$enddays = $monthdays[$_]
|
|
$currentmonthname = $monthnames[$_]
|
|
|
|
# setup start and end dates
|
|
[Datetime]$startDate = Get-Date -Year $TargetYear -Month $_ -Day 1
|
|
[Datetime]$endDate = Get-Date -Year $TargetYear -Month $_ -Day $enddays
|
|
|
|
# get data for the month
|
|
$MonthAcls = $TargetYearData | Where-Object {([Datetime]$_.CreationDate.trim() -ge $startDate -and [Datetime]$_.CreationDate.trim() -le $endDate)}
|
|
|
|
# Get acl count & and % of total
|
|
$MonthAclsCount = $MonthAcls | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclsCount -eq 0){
|
|
$MonthAclsCountP = 0;
|
|
}else{
|
|
$MonthAclsCountP = [math]::Round($MonthAclsCount/$HighestAclCountinMonth,4).tostring("P") -replace(" ","")
|
|
}
|
|
|
|
# Get share count
|
|
$MonthShareCount = $MonthAcls | select sharepath -Unique | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get computer count
|
|
$MonthComputerCount = $MonthAcls | select computername -Unique| Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get read count
|
|
$MonthAclReadCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*Read*") -or ($_.FileSystemRights -like "*Append*") } | Where-Object {($_.FileSystemRights -notlike "*GenericAll*") -and ($_.FileSystemRights -notlike "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclReadCount -eq 0){
|
|
$MonthAclReadCountP = 0;
|
|
$ReadDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:15px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclReadCountP = [math]::Round($MonthAclReadCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$ReadDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:15px;background-color:Orange;opacity: .25;`"></div>"
|
|
}
|
|
|
|
# Get write count
|
|
$MonthAclWriteCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclWriteCount -eq 0){
|
|
$MonthAclWriteCountP = 0;
|
|
$WriteDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:30px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclWriteCountP = [math]::Round($MonthAclWriteCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$WriteDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:30px;background-color:Orange;opacity: .5;`"></div>"
|
|
}
|
|
|
|
# Get hr count
|
|
$MonthAclHrCount = $MonthAcls | Where-Object {($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share')} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclHrCount -eq 0){
|
|
$MonthAclHrCountP = 0;
|
|
$HrDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:45px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclHrCountP = [math]::Round($MonthAclHrCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$HrDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:45px;background-color:Orange;opacity: 1;`"></div>"
|
|
}
|
|
|
|
# Debug
|
|
# write-host "$TargetYear - $_ - $startDate to $endDate - a=$MonthAclsCount s=$MonthShareCount c=$MonthComputerCount w=$MonthAclWriteCount r=$MonthAclReadCount hr=$MonthAclHrCount"
|
|
|
|
|
|
# build column
|
|
$HTMLMonth = @"
|
|
<div id="MonthItem" style="position: relative;float:left;padding-left:1px;padding-right:1px">
|
|
<div class="TimelineBarOutside" >
|
|
<div class="popwrapper" style="height:90px;position:relative;bottom:0;" >
|
|
<div class="TimelinePopup" style="position:absolute;">
|
|
<div class="TimelineMinicard">
|
|
<div class="TimelineMinicardtitle" align="center">
|
|
$currentmonthname $TargetYear<br>
|
|
</div>
|
|
<div class="TimelineMinicardcontainer" align="left">
|
|
|
|
<div style="margin-top:2px;padding-bottom:1px;">
|
|
<strong>Affected</strong><br>
|
|
Computers: $MonthComputerCount<br>
|
|
Shares: $MonthShareCount<Br>
|
|
ACLs: $MonthAclsCount<Br>
|
|
</div>
|
|
|
|
<div style="margin-top:1px;padding-bottom:1px;">
|
|
<strong>ACL Summary</strong><br>
|
|
Read: $MonthAclReadCount<br>
|
|
Write: $MonthAclWriteCount<br>
|
|
High-Risk: $MonthAclHrCount<br>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="TimelineBarInside" style="height:$MonthAclsCountP;width:100%;z-index: 0;"></div>
|
|
</div>
|
|
<div style="padding-bottom:15px;">
|
|
$ReadDot
|
|
$WriteDot
|
|
$HrDot
|
|
</div>
|
|
|
|
<div style="font-size:10;bottom:1;position:absolute;padding-left:3px;">$currentMonth</div>
|
|
</div>
|
|
</div>
|
|
|
|
"@
|
|
$HTMLMonth
|
|
|
|
} # END MONTH
|
|
|
|
|
|
$HTMLYearEnd = @"
|
|
</div>
|
|
<div id="bottom" align="center">
|
|
$TargetYear
|
|
</div>
|
|
</div>
|
|
</div>
|
|
"@
|
|
$HTMLYearEnd
|
|
|
|
} # END YEAR
|
|
|
|
|
|
$HTMLEND = @'
|
|
</div>
|
|
</div>
|
|
</div>
|
|
'@
|
|
|
|
$HTMLEND
|
|
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-CardLastAccess
|
|
# -------------------------------------------
|
|
function Get-CardLastAccess
|
|
{
|
|
[CmdletBinding()]
|
|
Param(
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Data table to parse. This should be the excessive privileges ACL table')]
|
|
$MyDataTable,
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Output file path.')]
|
|
[string]$OutFilePath = "Share-ACL-LastAccessDate-Monthly-Summary.csv"
|
|
)
|
|
|
|
# Get list of years for LastAccessDate - need to actual calculate all years to generate full timeline, even years with no data.
|
|
$ExcessivePrivsAclCount = $MyDataTable | measure | select count -expandproperty count
|
|
$ExcessivePrivsYears = $MyDataTable | select LastAccessDateYear -unique | Sort-Object LastAccessDateYear
|
|
$ExcessivePrivsYearsCount = $ExcessivePrivsYears | measure | select count -expandproperty count
|
|
$ExcessivePrivsYearsfirst = $ExcessivePrivsYears | select LastAccessDateYear -first 1 | select LastAccessDateYear -ExpandProperty LastAccessDateYear
|
|
$ExcessivePrivsYearsLast = $ExcessivePrivsYears | select LastAccessDateYear -last 1 | select LastAccessDateYear -ExpandProperty LastAccessDateYear
|
|
|
|
# Create data table to store summary data for chart scale
|
|
$ChartDataSummary = New-Object System.Data.DataTable
|
|
$null = $ChartDataSummary.Columns.Add("Year")
|
|
$null = $ChartDataSummary.Columns.Add("Month")
|
|
$null = $ChartDataSummary.Columns.Add("Computer")
|
|
$null = $ChartDataSummary.Columns.Add("Share")
|
|
$null = $ChartDataSummary.Columns.Add("ShareAcl")
|
|
$null = $ChartDataSummary.Columns.Add("AclRead")
|
|
$null = $ChartDataSummary.Columns.Add("AclWrite")
|
|
$null = $ChartDataSummary.Columns.Add("AclHR")
|
|
|
|
# ///////////////////////// Get bar chart ranges
|
|
|
|
# Year Loop
|
|
$ExcessivePrivsYears |
|
|
foreach {
|
|
|
|
$TargetYear = $_.LastAccessDateYear
|
|
$TargetYearData = $MyDataTable | where LastAccessDateYear -like "$TargetYear"
|
|
|
|
# Month looop
|
|
1..12 |
|
|
foreach {
|
|
|
|
# do last day of month look up here
|
|
$currentMonth = $_
|
|
$monthnames = @(0,"January","February","March","April","May","June","July","August","September","October","November","December")
|
|
$monthdays = @(0,31,28,31,30,31,30,31,31,30,31,30,31)
|
|
$enddays = $monthdays[$_]
|
|
$currentmonthname = $monthnames[$_]
|
|
|
|
# setup start and end dates
|
|
[Datetime]$startDate = Get-Date -Year $TargetYear -Month $_ -Day 1
|
|
[Datetime]$endDate = Get-Date -Year $TargetYear -Month $_ -Day $enddays
|
|
|
|
# get data for the month
|
|
$MonthAcls = $TargetYearData | Where-Object {([Datetime]$_.LastAccessDate.trim() -ge $startDate -and [Datetime]$_.LastAccessDate.trim() -le $endDate)}
|
|
|
|
# Get acl count & and % of total
|
|
$MonthAclsCount = $MonthAcls | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclsCount -eq 0){
|
|
$MonthAclsCountP = 0;
|
|
}else{
|
|
$MonthAclsCountP = [math]::Round($MonthAclsCount/$ExcessivePrivsAclCount,4).tostring("P") -replace(" ","")
|
|
}
|
|
|
|
# Get share count
|
|
$MonthShareCount = $MonthAcls | select sharepath -Unique | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get computer count
|
|
$MonthComputerCount = $MonthAcls | select computername -Unique| Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get read count
|
|
$MonthAclReadCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*Read*") -or ($_.FileSystemRights -like "*Append*") } | Where-Object {($_.FileSystemRights -notlike "*GenericAll*") -and ($_.FileSystemRights -notlike "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get write count
|
|
$MonthAclWriteCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get hr count
|
|
$MonthAclHrCount = $MonthAcls | Where-Object {($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share')} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Populate table
|
|
$null = $ChartDataSummary.Rows.Add("$TargetYear","$currentmonthname",$MonthComputerCount,$MonthShareCount,$MonthAclsCount,$MonthAclReadCount,$MonthAclWriteCount,$MonthAclHrCount)
|
|
|
|
# Export Results
|
|
$ChartDataSummary | Export-Csv -NoTypeInformation $OutFilePath
|
|
|
|
} #End Month
|
|
} # End year
|
|
|
|
# Get highest ShareAcl
|
|
$HighestAclCountinMonth = $ChartDataSummary | select shareacl | sort {[int]$_.shareacl} -Descending | select shareacl -First 1 -ExpandProperty shareacl
|
|
|
|
# Get highest AclRead
|
|
$HighestAclReadCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclRead} -Descending | Select-Object AclRead -First 1 -ExpandProperty AclRead
|
|
|
|
# Get highest AclWrite
|
|
$HighestAclWriteCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclWrite} -Descending | Select-Object AclWrite -First 1 -ExpandProperty AclWrite
|
|
|
|
# Get highest AclHR
|
|
$HighestAclHRCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclHR} -Descending | Select-Object AclHR -First 1 -ExpandProperty AclHR
|
|
|
|
# Get highst ACL type count
|
|
$TypeCounts = $HighestAclReadCountinMonth,$HighestAclWriteCountinMont,$HighestAclHRCountinMonth
|
|
$HighestTypeCount = $TypeCounts | Sort-Object {[int]$_} -Descending | select -First 1
|
|
|
|
|
|
# ///////////////////////// Get bar chart data
|
|
|
|
# Start Table
|
|
$HTML1 = @"
|
|
<div class="LargeCard">
|
|
<div class="LargeCardTitle">
|
|
Last Access Timeline<br>
|
|
<span class="LargeCardSubtitle2">for share ACLs configured with excessive privileges</span>
|
|
</div>
|
|
<div class="LargeCardContainer" align="center">
|
|
|
|
<div class="container" style="position: relative;float:left;bottom:0;left:0;height:195px;width:50px;">
|
|
<div style="top:5;position:absolute;color:#757575;width:100%;font-size:10;" align="right">$HighestAclCountinMonth <span style="color: #ccc">-</span> </div>
|
|
<div style="bottom: 98;position:absolute;color:#757575;border-bottom:1px solid #ccc;width:100%;font-size:10;padding-right:1px;" align="right">0 <span style="color: #ccc">-</span></div>
|
|
<div style="bottom:55;position:absolute;width:100%;font-size:10">
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">High Risk<br></div>
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">Write<br></div>
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">Read<br></div>
|
|
</div>
|
|
</div>
|
|
<div class="TimelineChart" Style="grid-template-columns: 1px repeat($ExcessivePrivsYearsCount, 204px) 1px;">
|
|
"@
|
|
|
|
$HTML1
|
|
|
|
# Year Loop
|
|
$ExcessivePrivsYears |
|
|
foreach {
|
|
|
|
$TargetYear = $_.LastAccessDateYear
|
|
$TargetYearData = $MyDataTable | where LastAccessDateYear -like "$TargetYear"
|
|
|
|
# Start Year
|
|
$HTMLYearStart = @'
|
|
<div class="YearItem" >
|
|
<div id="YearWrapper" style="float:left;background-color:#F2F3F4;">
|
|
<div id="MonthsWrapper" style="position: relative;float:left">
|
|
|
|
'@
|
|
$HTMLYearStart
|
|
|
|
# Month looop
|
|
1..12 |
|
|
foreach {
|
|
|
|
# do last day of month look up here
|
|
$currentMonth = $_
|
|
$monthnames = @(0,"January","February","March","April","May","June","July","August","September","October","November","December")
|
|
$monthdays = @(0,31,28,31,30,31,30,31,31,30,31,30,31)
|
|
$enddays = $monthdays[$_]
|
|
$currentmonthname = $monthnames[$_]
|
|
|
|
# setup start and end dates
|
|
[Datetime]$startDate = Get-Date -Year $TargetYear -Month $_ -Day 1
|
|
[Datetime]$endDate = Get-Date -Year $TargetYear -Month $_ -Day $enddays
|
|
|
|
# get data for the month
|
|
$MonthAcls = $TargetYearData | Where-Object {([Datetime]$_.LastAccessDate.trim() -ge $startDate -and [Datetime]$_.LastAccessDate.trim() -le $endDate)}
|
|
|
|
# Get acl count & and % of total
|
|
$MonthAclsCount = $MonthAcls | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclsCount -eq 0){
|
|
$MonthAclsCountP = 0;
|
|
}else{
|
|
$MonthAclsCountP = [math]::Round($MonthAclsCount/$HighestAclCountinMonth,4).tostring("P") -replace(" ","")
|
|
}
|
|
|
|
# Get share count
|
|
$MonthShareCount = $MonthAcls | select sharepath -Unique | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get computer count
|
|
$MonthComputerCount = $MonthAcls | select computername -Unique| Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get read count
|
|
$MonthAclReadCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*Read*") -or ($_.FileSystemRights -like "*Append*") } | Where-Object {($_.FileSystemRights -notlike "*GenericAll*") -and ($_.FileSystemRights -notlike "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclReadCount -eq 0){
|
|
$MonthAclReadCountP = 0;
|
|
$ReadDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:15px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclReadCountP = [math]::Round($MonthAclReadCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$ReadDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:15px;background-color:Orange;opacity: .25;`"></div>"
|
|
}
|
|
|
|
# Get write count
|
|
$MonthAclWriteCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclWriteCount -eq 0){
|
|
$MonthAclWriteCountP = 0;
|
|
$WriteDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:30px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclWriteCountP = [math]::Round($MonthAclWriteCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$WriteDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:30px;background-color:Orange;opacity: .5;`"></div>"
|
|
}
|
|
|
|
# Get hr count
|
|
$MonthAclHrCount = $MonthAcls | Where-Object {($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share')} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclHrCount -eq 0){
|
|
$MonthAclHrCountP = 0;
|
|
$HrDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:45px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclHrCountP = [math]::Round($MonthAclHrCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$HrDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:45px;background-color:Orange;opacity: 1;`"></div>"
|
|
}
|
|
|
|
# Debug
|
|
# write-host "$TargetYear - $_ - $startDate to $endDate - a=$MonthAclsCount s=$MonthShareCount c=$MonthComputerCount w=$MonthAclWriteCount r=$MonthAclReadCount hr=$MonthAclHrCount"
|
|
|
|
|
|
# build column
|
|
$HTMLMonth = @"
|
|
<div id="MonthItem" style="position: relative;float:left;padding-left:1px;padding-right:1px">
|
|
<div class="TimelineBarOutside" >
|
|
<div class="popwrapper" style="height:90px;position:relative;bottom:0;" >
|
|
<div class="TimelinePopup" style="position:absolute;">
|
|
<div class="TimelineMinicard">
|
|
<div class="TimelineMinicardtitle" align="center">
|
|
$currentmonthname $TargetYear<br>
|
|
</div>
|
|
<div class="TimelineMinicardcontainer" align="left">
|
|
|
|
<div style="margin-top:2px;padding-bottom:1px;">
|
|
<strong>Affected</strong><br>
|
|
Computers: $MonthComputerCount<br>
|
|
Shares: $MonthShareCount<Br>
|
|
ACLs: $MonthAclsCount<Br>
|
|
</div>
|
|
|
|
<div style="margin-top:1px;padding-bottom:1px;">
|
|
<strong>ACL Summary</strong><br>
|
|
Read: $MonthAclReadCount<br>
|
|
Write: $MonthAclWriteCount<br>
|
|
High-Risk: $MonthAclHrCount<br>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="TimelineBarInside" style="height:$MonthAclsCountP;width:100%;z-index: 0;"></div>
|
|
</div>
|
|
<div style="padding-bottom:15px;">
|
|
$ReadDot
|
|
$WriteDot
|
|
$HrDot
|
|
</div>
|
|
|
|
<div style="font-size:10;bottom:1;position:absolute;padding-left:3px;">$currentMonth</div>
|
|
</div>
|
|
</div>
|
|
|
|
"@
|
|
$HTMLMonth
|
|
|
|
} # END MONTH
|
|
|
|
|
|
$HTMLYearEnd = @"
|
|
</div>
|
|
<div id="bottom" align="center">
|
|
$TargetYear
|
|
</div>
|
|
</div>
|
|
</div>
|
|
"@
|
|
$HTMLYearEnd
|
|
|
|
} # END YEAR
|
|
|
|
|
|
$HTMLEND = @'
|
|
</div>
|
|
</div>
|
|
</div>
|
|
'@
|
|
|
|
$HTMLEND
|
|
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-CardLastModified
|
|
# -------------------------------------------
|
|
function Get-CardLastModified
|
|
{
|
|
[CmdletBinding()]
|
|
Param(
|
|
[Parameter(Mandatory = $true,
|
|
HelpMessage = 'Data table to parse. This should be the excessive privileges ACL table.')]
|
|
$MyDataTable,
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Output file path.')]
|
|
[string]$OutFilePath = "Share-ACL-LastModifiedDate-Monthly-Summary.csv"
|
|
)
|
|
|
|
# Get list of years for LastModifiedDate - need to actual calculate all years to generate full timeline, even years with no data.
|
|
$ExcessivePrivsAclCount = $MyDataTable | measure | select count -expandproperty count
|
|
$ExcessivePrivsYears = $MyDataTable | select LastModifiedDateYear -unique | Sort-Object LastModifiedDateYear
|
|
$ExcessivePrivsYearsCount = $ExcessivePrivsYears | measure | select count -expandproperty count
|
|
$ExcessivePrivsYearsfirst = $ExcessivePrivsYears | select LastModifiedDateYear -first 1 | select LastModifiedDateYear -ExpandProperty LastModifiedDateYear
|
|
$ExcessivePrivsYearsLast = $ExcessivePrivsYears | select LastModifiedDateYear -last 1 | select LastModifiedDateYear -ExpandProperty LastModifiedDateYear
|
|
|
|
# Create data table to store summary data for chart scale
|
|
$ChartDataSummary = New-Object System.Data.DataTable
|
|
$null = $ChartDataSummary.Columns.Add("Year")
|
|
$null = $ChartDataSummary.Columns.Add("Month")
|
|
$null = $ChartDataSummary.Columns.Add("Computer")
|
|
$null = $ChartDataSummary.Columns.Add("Share")
|
|
$null = $ChartDataSummary.Columns.Add("ShareAcl")
|
|
$null = $ChartDataSummary.Columns.Add("AclRead")
|
|
$null = $ChartDataSummary.Columns.Add("AclWrite")
|
|
$null = $ChartDataSummary.Columns.Add("AclHR")
|
|
|
|
# ///////////////////////// Get bar chart ranges
|
|
|
|
# Year Loop
|
|
$ExcessivePrivsYears |
|
|
foreach {
|
|
|
|
$TargetYear = $_.LastModifiedDateYear
|
|
$TargetYearData = $MyDataTable | where LastModifiedDateYear -like "$TargetYear"
|
|
|
|
# Month looop
|
|
1..12 |
|
|
foreach {
|
|
|
|
# do last day of month look up here
|
|
$currentMonth = $_
|
|
$monthnames = @(0,"January","February","March","April","May","June","July","August","September","October","November","December")
|
|
$monthdays = @(0,31,28,31,30,31,30,31,31,30,31,30,31)
|
|
$enddays = $monthdays[$_]
|
|
$currentmonthname = $monthnames[$_]
|
|
|
|
# setup start and end dates
|
|
[Datetime]$startDate = Get-Date -Year $TargetYear -Month $_ -Day 1
|
|
[Datetime]$endDate = Get-Date -Year $TargetYear -Month $_ -Day $enddays
|
|
|
|
# get data for the month
|
|
$MonthAcls = $TargetYearData | Where-Object {([Datetime]$_.LastModifiedDate.trim() -ge $startDate -and [Datetime]$_.LastModifiedDate.trim() -le $endDate)}
|
|
|
|
# Get acl count & and % of total
|
|
$MonthAclsCount = $MonthAcls | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclsCount -eq 0){
|
|
$MonthAclsCountP = 0;
|
|
}else{
|
|
$MonthAclsCountP = [math]::Round($MonthAclsCount/$ExcessivePrivsAclCount,4).tostring("P") -replace(" ","")
|
|
}
|
|
|
|
# Get share count
|
|
$MonthShareCount = $MonthAcls | select sharepath -Unique | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get computer count
|
|
$MonthComputerCount = $MonthAcls | select computername -Unique| Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get read count
|
|
$MonthAclReadCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*Read*") -or ($_.FileSystemRights -like "*Append*") } | Where-Object {($_.FileSystemRights -notlike "*GenericAll*") -and ($_.FileSystemRights -notlike "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get write count
|
|
$MonthAclWriteCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get hr count
|
|
$MonthAclHrCount = $MonthAcls | Where-Object {($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share')} | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Populate table
|
|
$null = $ChartDataSummary.Rows.Add("$TargetYear","$currentmonthname",$MonthComputerCount,$MonthShareCount,$MonthAclsCount,$MonthAclReadCount,$MonthAclWriteCount,$MonthAclHrCount)
|
|
|
|
# Export Results
|
|
$ChartDataSummary | Export-Csv -NoTypeInformation $OutFilePath
|
|
|
|
} #End Month Loop
|
|
} # End ear loop
|
|
|
|
# Get highest ShareAcl
|
|
$HighestAclCountinMonth = $ChartDataSummary | select shareacl | sort {[int]$_.shareacl} -Descending | select shareacl -First 1 -ExpandProperty shareacl
|
|
|
|
# Get highest AclRead
|
|
$HighestAclReadCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclRead} -Descending | Select-Object AclRead -First 1 -ExpandProperty AclRead
|
|
|
|
# Get highest AclWrite
|
|
$HighestAclWriteCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclWrite} -Descending | Select-Object AclWrite -First 1 -ExpandProperty AclWrite
|
|
|
|
# Get highest AclHR
|
|
$HighestAclHRCountinMonth = $ChartDataSummary | Sort-Object {[int]$_.AclHR} -Descending | Select-Object AclHR -First 1 -ExpandProperty AclHR
|
|
|
|
# Get highst ACL type count
|
|
$TypeCounts = $HighestAclReadCountinMonth,$HighestAclWriteCountinMont,$HighestAclHRCountinMonth
|
|
$HighestTypeCount = $TypeCounts | Sort-Object {[int]$_} -Descending | select -First 1
|
|
|
|
|
|
# ///////////////////////// Get bar chart data
|
|
|
|
# Start Table
|
|
$HTML1 = @"
|
|
<div class="LargeCard">
|
|
<div class="LargeCardTitle">
|
|
Last Write Timeline<br>
|
|
<span class="LargeCardSubtitle2">for share ACLs configured with excessive privileges</span>
|
|
</div>
|
|
<div class="LargeCardContainer" align="center">
|
|
|
|
<div class="container" style="position: relative;float:left;bottom:0;left:0;height:195px;width:50px;">
|
|
<div style="top:5;position:absolute;color:#757575;width:100%;font-size:10;" align="right">$HighestAclCountinMonth <span style="color: #ccc">-</span> </div>
|
|
<div style="bottom: 98;position:absolute;color:#757575;border-bottom:1px solid #ccc;width:100%;font-size:10;padding-right:1px;" align="right">0 <span style="color: #ccc">-</span></div>
|
|
<div style="bottom:55;position:absolute;width:100%;font-size:10">
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">High Risk<br></div>
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">Write<br></div>
|
|
<div width="100%" align="right" style="padding-right:3px;color:#757575">Read<br></div>
|
|
</div>
|
|
</div>
|
|
<div class="TimelineChart" Style="grid-template-columns: 1px repeat($ExcessivePrivsYearsCount, 204px) 1px;">
|
|
"@
|
|
|
|
$HTML1
|
|
|
|
# Year Loop
|
|
$ExcessivePrivsYears |
|
|
foreach {
|
|
|
|
$TargetYear = $_.LastModifiedDateYear
|
|
$TargetYearData = $MyDataTable | where LastModifiedDateYear -like "$TargetYear"
|
|
|
|
# Start Year
|
|
$HTMLYearStart = @'
|
|
<div class="YearItem" >
|
|
<div id="YearWrapper" style="float:left;background-color:#F2F3F4;">
|
|
<div id="MonthsWrapper" style="position: relative;float:left">
|
|
|
|
'@
|
|
$HTMLYearStart
|
|
|
|
# Month looop
|
|
1..12 |
|
|
foreach {
|
|
|
|
# do last day of month look up here
|
|
$currentMonth = $_
|
|
$monthnames = @(0,"January","February","March","April","May","June","July","August","September","October","November","December")
|
|
$monthdays = @(0,31,28,31,30,31,30,31,31,30,31,30,31)
|
|
$enddays = $monthdays[$_]
|
|
$currentmonthname = $monthnames[$_]
|
|
|
|
# setup start and end dates
|
|
[Datetime]$startDate = Get-Date -Year $TargetYear -Month $_ -Day 1
|
|
[Datetime]$endDate = Get-Date -Year $TargetYear -Month $_ -Day $enddays
|
|
|
|
# get data for the month
|
|
$MonthAcls = $TargetYearData | Where-Object {([Datetime]$_.LastModifiedDate.trim() -ge $startDate -and [Datetime]$_.LastModifiedDate.trim() -le $endDate)}
|
|
|
|
# Get acl count & and % of total
|
|
$MonthAclsCount = $MonthAcls | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclsCount -eq 0){
|
|
$MonthAclsCountP = 0;
|
|
}else{
|
|
$MonthAclsCountP = [math]::Round($MonthAclsCount/$HighestAclCountinMonth,4).tostring("P") -replace(" ","")
|
|
}
|
|
|
|
# Get share count
|
|
$MonthShareCount = $MonthAcls | select sharepath -Unique | Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get computer count
|
|
$MonthComputerCount = $MonthAcls | select computername -Unique| Measure-Object | select count -ExpandProperty count
|
|
|
|
# Get read count
|
|
$MonthAclReadCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*Read*") -or ($_.FileSystemRights -like "*Append*") } | Where-Object {($_.FileSystemRights -notlike "*GenericAll*") -and ($_.FileSystemRights -notlike "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclReadCount -eq 0){
|
|
$MonthAclReadCountP = 0;
|
|
$ReadDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:15px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclReadCountP = [math]::Round($MonthAclReadCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$ReadDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:15px;background-color:Orange;opacity: .25;`"></div>"
|
|
}
|
|
|
|
# Get write count
|
|
$MonthAclWriteCount = $MonthAcls | Where-Object {($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*")} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclWriteCount -eq 0){
|
|
$MonthAclWriteCountP = 0;
|
|
$WriteDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:30px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclWriteCountP = [math]::Round($MonthAclWriteCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$WriteDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:30px;background-color:Orange;opacity: .5;`"></div>"
|
|
}
|
|
|
|
# Get hr count
|
|
$MonthAclHrCount = $MonthAcls | Where-Object {($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share')} | Measure-Object | select count -ExpandProperty count
|
|
if($MonthAclHrCount -eq 0){
|
|
$MonthAclHrCountP = 0;
|
|
$HrDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:45px;background-color:gray;`"></div>"
|
|
}else{
|
|
$MonthAclHrCountP = [math]::Round($MonthAclHrCount/$HighestTypeCount,4).tostring("P") -replace(" ","")
|
|
$HrDot = "<div class=`"TimelineDot`" style=`"border:1px solid #F2F3F4;bottom:45px;background-color:Orange;opacity: 1;`"></div>"
|
|
}
|
|
|
|
# Debug
|
|
# write-host "$TargetYear - $_ - $startDate to $endDate - a=$MonthAclsCount s=$MonthShareCount c=$MonthComputerCount w=$MonthAclWriteCount r=$MonthAclReadCount hr=$MonthAclHrCount"
|
|
|
|
|
|
# build column
|
|
$HTMLMonth = @"
|
|
<div id="MonthItem" style="position: relative;float:left;padding-left:1px;padding-right:1px">
|
|
<div class="TimelineBarOutside" >
|
|
<div class="popwrapper" style="height:90px;position:relative;bottom:0;" >
|
|
<div class="TimelinePopup" style="position:absolute;">
|
|
<div class="TimelineMinicard">
|
|
<div class="TimelineMinicardtitle" align="center">
|
|
$currentmonthname $TargetYear<br>
|
|
</div>
|
|
<div class="TimelineMinicardcontainer" align="left">
|
|
|
|
<div style="margin-top:2px;padding-bottom:1px;">
|
|
<strong>Affected</strong><br>
|
|
Computers: $MonthComputerCount<br>
|
|
Shares: $MonthShareCount<Br>
|
|
ACLs: $MonthAclsCount<Br>
|
|
</div>
|
|
|
|
<div style="margin-top:1px;padding-bottom:1px;">
|
|
<strong>ACL Summary</strong><br>
|
|
Read: $MonthAclReadCount<br>
|
|
Write: $MonthAclWriteCount<br>
|
|
High-Risk: $MonthAclHrCount<br>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="TimelineBarInside" style="height:$MonthAclsCountP;width:100%;z-index: 0;"></div>
|
|
</div>
|
|
<div style="padding-bottom:15px;">
|
|
$ReadDot
|
|
$WriteDot
|
|
$HrDot
|
|
</div>
|
|
|
|
<div style="font-size:10;bottom:1;position:absolute;padding-left:3px;">$currentMonth</div>
|
|
</div>
|
|
</div>
|
|
|
|
"@
|
|
$HTMLMonth
|
|
|
|
} #END MONTH LOOP
|
|
|
|
|
|
$HTMLYearEnd = @"
|
|
</div>
|
|
<div id="bottom" align="center">
|
|
$TargetYear
|
|
</div>
|
|
</div>
|
|
</div>
|
|
"@
|
|
$HTMLYearEnd
|
|
|
|
} # End YEAR LOOP
|
|
|
|
|
|
# End Page
|
|
|
|
$HTMLEND = @'
|
|
</div>
|
|
</div>
|
|
</div>
|
|
'@
|
|
|
|
$HTMLEND
|
|
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-PercentDisplay
|
|
# -------------------------------------------
|
|
function Get-PercentDisplay
|
|
{
|
|
param (
|
|
$TargetCount,
|
|
$FullCount
|
|
)
|
|
|
|
if($FullCount -ne 0){
|
|
$Percent = [math]::Round($TargetCount/$FullCount,4)
|
|
$PercentString = $Percent.tostring("P") -replace(" ","")
|
|
$PercentBarVal = ($Percent *2).tostring("P") -replace(" %","px")
|
|
}
|
|
|
|
# Return object with all counts
|
|
$TheCounts = new-object psobject
|
|
$TheCounts | add-member Noteproperty PercentString $PercentString
|
|
$TheCounts | add-member Noteproperty PercentBarVal $PercentBarVal
|
|
$TheCounts
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-ExPrivSumData
|
|
# -------------------------------------------
|
|
# get the computer, share, and acl summary data for a provided data table
|
|
function Get-ExPrivSumData
|
|
{
|
|
param (
|
|
$DataTable,
|
|
$AllComputerCount,
|
|
$AllShareCount,
|
|
$AllAclCount
|
|
)
|
|
|
|
# Get acl counts
|
|
$UserAcls = $DataTable
|
|
$UserAclsCount = $UserAcls | measure | select count -ExpandProperty count
|
|
$UserAclsPercent = [math]::Round($UserAclsCount/$AllAclCount,4)
|
|
$UserAclsPercentString = $UserAclsPercent.tostring("P") -replace(" ","")
|
|
$UserAclsPercentBarVal = ($UserAclsPercent *2).tostring("P") -replace(" %","px")
|
|
$UserAclsPercentBarCode = "<span class=`"dashboardsub2`">$UserAclsPercentString ($UserAclsCount of $AllAclCount)</span><br><div class=`"cardbarouter`"><div class=`"cardbarinside`" style=`"width: $UserAclsPercentString;`"></div></div>"
|
|
|
|
# Get share counts
|
|
$UserShare = $UserAcls | Select-Object SharePath -Unique
|
|
$UserShareCount = $UserShare | measure | select count -ExpandProperty count
|
|
$UserSharePercent = [math]::Round($UserShareCount/$AllShareCount,4)
|
|
$UserSharePercentString = $UserSharePercent.tostring("P") -replace(" ","")
|
|
$UserSharePercentBarVal = ($UserSharePercent *2).tostring("P") -replace(" %","px")
|
|
$UserSharePercentBarCode = "<span class=`"dashboardsub2`">$UserSharePercentString ($UserShareCount of $AllShareCount)</span><br><div class=`"cardbarouter`"><div class=`"cardbarinside`" style=`"width: $UserSharePercentString;`"></div></div>"
|
|
|
|
# Get computer counts
|
|
$UserComputer = $UserAcls | Select-Object ComputerName -Unique
|
|
$UserComputerCount = $UserComputer | measure | select count -ExpandProperty count
|
|
$UserComputerPercent = [math]::Round($UserComputerCount/$AllComputerCount,4)
|
|
$UserComputerPercentString = $UserComputerPercent.tostring("P") -replace(" ","")
|
|
$UserComputerPercentBarVal = ($UserComputerPercent *2).tostring("P") -replace(" %","px")
|
|
$UserComputerPercentBarCode = "<span class=`"dashboardsub2`">$UserComputerPercentString ($UserComputerCount of $AllComputerCount)</span><br><div class=`"cardbarouter`"><div class=`"cardbarinside`" style=`"width: $UserComputerPercentString;`"></div></div>"
|
|
|
|
# Return object with all counts
|
|
$TheCounts = new-object psobject
|
|
$TheCounts | add-member Noteproperty ComputerBar $UserComputerPercentBarCode
|
|
$TheCounts | add-member Noteproperty ShareBar $UserSharePercentBarCode
|
|
$TheCounts | add-member Noteproperty AclBar $UserAclsPercentBarCode
|
|
$TheCounts
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-GroupOwnerBar
|
|
# -------------------------------------------
|
|
function Get-GroupOwnerBar
|
|
{
|
|
param (
|
|
$DataTable,
|
|
$Name,
|
|
$AllComputerCount,
|
|
$AllShareCount,
|
|
$AllAclCount
|
|
)
|
|
|
|
# Get acl counts
|
|
$UserAcls = $DataTable | Where ShareOwner -like "$Name" | Select-Object ComputerName,ShareOwner,SharePath,FileSystemRights
|
|
$UserAclsCount = $UserAcls | measure | select count -ExpandProperty count
|
|
$UserAclsPercent = [math]::Round($UserAclsCount/$AllAclCount,4)
|
|
$UserAclsPercentString = $UserAclsPercent.tostring("P") -replace(" ","")
|
|
$UserAclsPercentBarVal = ($UserAclsPercent *2).tostring("P") -replace(" %","px")
|
|
$UserAclsPercentBarCode = "<span class=`"dashboardsub2`">$UserAclsPercentString ($UserAclsCount of $AllAclCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserAclsPercentString;`"></div></div>"
|
|
|
|
# Get share counts
|
|
$UserShare = $UserAcls | Select-Object SharePath -Unique
|
|
$UserShareCount = $UserShare | measure | select count -ExpandProperty count
|
|
$UserSharePercent = [math]::Round($UserShareCount/$AllShareCount,4)
|
|
$UserSharePercentString = $UserSharePercent.tostring("P") -replace(" ","")
|
|
$UserSharePercentBarVal = ($UserSharePercent *2).tostring("P") -replace(" %","px")
|
|
$UserSharePercentBarCode = "<span class=`"dashboardsub2`">$UserSharePercentString ($UserShareCount of $AllShareCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserSharePercentString;`"></div></div>"
|
|
|
|
# Get computer counts
|
|
$UserComputer = $UserAcls | Select-Object ComputerName -Unique
|
|
$UserComputerCount = $UserComputer | measure | select count -ExpandProperty count
|
|
$UserComputerPercent = [math]::Round($UserComputerCount/$AllComputerCount,4)
|
|
$UserComputerPercentString = $UserComputerPercent.tostring("P") -replace(" ","")
|
|
$UserComputerPercentBarVal = ($UserComputerPercent *2).tostring("P") -replace(" %","px")
|
|
$UserComputerPercentBarCode = "<span class=`"dashboardsub2`">$UserComputerPercentString ($UserComputerCount of $AllComputerCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserComputerPercentString;`"></div></div>"
|
|
|
|
# Return object with all counts
|
|
$TheCounts = new-object psobject
|
|
$TheCounts | add-member Noteproperty ComputerBar $UserComputerPercentBarCode
|
|
$TheCounts | add-member Noteproperty ShareBar $UserSharePercentBarCode
|
|
$TheCounts | add-member Noteproperty AclBar $UserAclsPercentBarCode
|
|
$TheCounts
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-GroupNameBar
|
|
# -------------------------------------------
|
|
function Get-GroupNameBar
|
|
{
|
|
param (
|
|
$DataTable,
|
|
$Name,
|
|
$AllComputerCount,
|
|
$AllShareCount,
|
|
$AllAclCount
|
|
)
|
|
|
|
# Get acl counts
|
|
$UserAcls = $DataTable | Where ShareName -like "$Name" | Select-Object ComputerName, ShareName, SharePath, FileSystemRights
|
|
$UserAclsCount = $UserAcls | measure | select count -ExpandProperty count
|
|
$UserAclsPercent = [math]::Round($UserAclsCount/$AllAclCount,4)
|
|
$UserAclsPercentString = $UserAclsPercent.tostring("P") -replace(" ","")
|
|
$UserAclsPercentBarVal = ($UserAclsPercent *2).tostring("P") -replace(" %","px")
|
|
$UserAclsPercentBarCode = "<span class=`"dashboardsub2`">$UserAclsPercentString ($UserAclsCount of $AllAclCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserAclsPercentString;`"></div></div>"
|
|
|
|
# Get share counts
|
|
$UserShare = $UserAcls | Select-Object SharePath -Unique
|
|
$UserShareCount = $UserShare | measure | select count -ExpandProperty count
|
|
$UserSharePercent = [math]::Round($UserShareCount/$AllShareCount,4)
|
|
$UserSharePercentString = $UserSharePercent.tostring("P") -replace(" ","")
|
|
$UserSharePercentBarVal = ($UserSharePercent *2).tostring("P") -replace(" %","px")
|
|
$UserSharePercentBarCode = "<span class=`"dashboardsub2`">$UserSharePercentString ($UserShareCount of $AllShareCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserSharePercentString;`"></div></div>"
|
|
|
|
# Get computer counts
|
|
$UserComputer = $UserAcls | Select-Object ComputerName -Unique
|
|
$UserComputerCount = $UserComputer | measure | select count -ExpandProperty count
|
|
$UserComputerPercent = [math]::Round($UserComputerCount/$AllComputerCount,4)
|
|
$UserComputerPercentString = $UserComputerPercent.tostring("P") -replace(" ","")
|
|
$UserComputerPercentBarVal = ($UserComputerPercent *2).tostring("P") -replace(" %","px")
|
|
$UserComputerPercentBarCode = "<span class=`"dashboardsub2`">$UserComputerPercentString ($UserComputerCount of $AllComputerCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserComputerPercentString;`"></div></div>"
|
|
|
|
# Return object with all counts
|
|
$TheCounts = new-object psobject
|
|
$TheCounts | add-member Noteproperty ComputerBar $UserComputerPercentBarCode
|
|
$TheCounts | add-member Noteproperty ShareBar $UserSharePercentBarCode
|
|
$TheCounts | add-member Noteproperty AclBar $UserAclsPercentBarCode
|
|
$TheCounts
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-GroupNameNoBar
|
|
# -------------------------------------------
|
|
function Get-GroupNameNoBar
|
|
{
|
|
param (
|
|
$DataTable,
|
|
$Name,
|
|
$AllComputerCount,
|
|
$AllShareCount,
|
|
$AllAclCount
|
|
)
|
|
|
|
# Get acl counts
|
|
$UserAcls = $DataTable | Where ShareName -like "$Name" | Select-Object ComputerName, ShareName, SharePath, FileSystemRights
|
|
$UserAclsCount = $UserAcls | measure | select count -ExpandProperty count
|
|
$UserAclsPercent = [math]::Round($UserAclsCount/$AllAclCount,4)
|
|
$UserAclsPercentString = $UserAclsPercent.tostring("P") -replace(" ","")
|
|
$UserAclsPercentBarVal = ($UserAclsPercent *2).tostring("P") -replace(" %","px")
|
|
$UserAclsPercentBarCode = @"
|
|
<button class="collapsible" style="font-size: 10px;"><strong>$UserAclsCount</strong> Acls</button>
|
|
<div class="content">
|
|
<div class="subexpandnocolor" style="font-size: 10px; background-color: none;" >
|
|
$UserAclsCount of $AllAclCount ($UserAclsPercentString)
|
|
</div>
|
|
</div>
|
|
"@
|
|
|
|
# Get share counts
|
|
$UserShare = $UserAcls | Select-Object SharePath -Unique
|
|
$UserShareCount = $UserShare | measure | select count -ExpandProperty count
|
|
$UserSharePercent = [math]::Round($UserShareCount/$AllShareCount,4)
|
|
$UserSharePercentString = $UserSharePercent.tostring("P") -replace(" ","")
|
|
$UserSharePercentBarVal = ($UserSharePercent *2).tostring("P") -replace(" %","px")
|
|
$UserSharePercentBarCode = @"
|
|
<button class="collapsible" style="font-size: 10px;"><strong>$UserShareCount</strong> Shares</button>
|
|
<div class="content">
|
|
<div class="subexpandnocolor" style="font-size: 10px; background-color: none;" >
|
|
$UserShareCount of $AllShareCount ($UserSharePercentString)
|
|
</div>
|
|
</div>
|
|
"@
|
|
|
|
# Get computer counts
|
|
$UserComputer = $UserAcls | Select-Object ComputerName -Unique
|
|
$UserComputerCount = $UserComputer | measure | select count -ExpandProperty count
|
|
$UserComputerPercent = [math]::Round($UserComputerCount/$AllComputerCount,4)
|
|
$UserComputerPercentString = $UserComputerPercent.tostring("P") -replace(" ","")
|
|
$UserComputerPercentBarVal = ($UserComputerPercent *2).tostring("P") -replace(" %","px")
|
|
$UserComputerPercentBarCode = @"
|
|
<button class="collapsible" style="font-size: 10px;"><strong>$UserComputerCount</strong> Computers</button>
|
|
<div class="content">
|
|
<div class="subexpandnocolor" style="font-size: 10px; background-color: none;" >
|
|
$UserComputerCount of $AllComputerCount ($UserComputerPercentString)
|
|
</div>
|
|
</div>
|
|
"@
|
|
|
|
# Return object with all counts
|
|
$TheCounts = new-object psobject
|
|
$TheCounts | add-member Noteproperty ComputerBar $UserComputerPercentBarCode
|
|
$TheCounts | add-member Noteproperty ShareBar $UserSharePercentBarCode
|
|
$TheCounts | add-member Noteproperty AclBar $UserAclsPercentBarCode
|
|
$TheCounts
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-GroupFileBar
|
|
# -------------------------------------------
|
|
function Get-GroupFileBar
|
|
{
|
|
param (
|
|
$DataTable,
|
|
$Name,
|
|
$AllComputerCount,
|
|
$AllShareCount,
|
|
$AllAclCount
|
|
)
|
|
|
|
# Get acl counts
|
|
$UserAcls = $DataTable | Where FileListGroup -like "$Name" | Select-Object ComputerName, ShareName, SharePath, FileSystemRights, FileCount, FileList
|
|
$FolderInfo = $UserAcls | select FileCount, FileList -First 1
|
|
$FileCount = $FolderInfo.FileCount
|
|
$FileList = $FolderInfo.FileList
|
|
$UserAclsCount = $UserAcls | measure | select count -ExpandProperty count
|
|
$UserAclsPercent = [math]::Round($UserAclsCount/$AllAclCount,4)
|
|
$UserAclsPercentString = $UserAclsPercent.tostring("P") -replace(" ","")
|
|
$UserAclsPercentBarVal = ($UserAclsPercent *2).tostring("P") -replace(" %","px")
|
|
$UserAclsPercentBarCode = "<span class=`"dashboardsub2`">$UserAclsPercentString ($UserAclsCount of $AllAclCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserAclsPercentString;`"></div></div>"
|
|
|
|
# Get share counts
|
|
$UserShare = $UserAcls | Select-Object SharePath -Unique
|
|
$UserShareCount = $UserShare | measure | select count -ExpandProperty count
|
|
$UserSharePercent = [math]::Round($UserShareCount/$AllShareCount,4)
|
|
$UserSharePercentString = $UserSharePercent.tostring("P") -replace(" ","")
|
|
$UserSharePercentBarVal = ($UserSharePercent *2).tostring("P") -replace(" %","px")
|
|
$UserSharePercentBarCode = "<span class=`"dashboardsub2`">$UserSharePercentString ($UserShareCount of $AllShareCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserSharePercentString;`"></div></div>"
|
|
|
|
# Get computer counts
|
|
$UserComputer = $UserAcls | Select-Object ComputerName -Unique
|
|
$UserComputerCount = $UserComputer | measure | select count -ExpandProperty count
|
|
if($AllComputerCount -ne 0){
|
|
$UserComputerPercent = [math]::Round($UserComputerCount/$AllComputerCount,4)
|
|
}
|
|
$UserComputerPercentString = $UserComputerPercent.tostring("P") -replace(" ","")
|
|
$UserComputerPercentBarVal = ($UserComputerPercent *2).tostring("P") -replace(" %","px")
|
|
$UserComputerPercentBarCode = "<span class=`"dashboardsub2`">$UserComputerPercentString ($UserComputerCount of $AllComputerCount)</span><br><div class=`"divbarDomain`"><div class=`"divbarDomainInside`" style=`"width: $UserComputerPercentString;`"></div></div>"
|
|
|
|
# Return object with all counts
|
|
$TheCounts = new-object psobject
|
|
$TheCounts | add-member Noteproperty ComputerBar $UserComputerPercentBarCode
|
|
$TheCounts | add-member Noteproperty ShareBar $UserSharePercentBarCode
|
|
$TheCounts | add-member Noteproperty AclBar $UserAclsPercentBarCode
|
|
$TheCounts | add-member Noteproperty FileCount $FileCount
|
|
$TheCounts | add-member Noteproperty FileList $FileList
|
|
$TheCounts | add-member Noteproperty ShareCount $UserShareCount
|
|
$TheCounts
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-UserAceCounts
|
|
# -------------------------------------------
|
|
function Get-UserAceCounts
|
|
{
|
|
param (
|
|
$DataTable,
|
|
$UserName
|
|
)
|
|
|
|
# Get acl counts
|
|
$UserAcls = $DataTable | Where IdentityReference -like "$UserName" | Select-Object ComputerName, ShareName, SharePath, FileSystemRights
|
|
$UserAclsCount = $UserAcls | measure | select count -ExpandProperty count
|
|
|
|
# Get share counts
|
|
$UserShare = $UserAcls | Select-Object SharePath -Unique
|
|
$UserShareCount = $UserShare | measure | select count -ExpandProperty count
|
|
|
|
# Get computer counts
|
|
$UserComputer = $UserAcls | Select-Object ComputerName -Unique
|
|
$UserComputerCount = $UserComputer | measure | select count -ExpandProperty count
|
|
|
|
# Get read counts
|
|
$UserReadAcl = $UserAcls |
|
|
Foreach {
|
|
|
|
if(($_.FileSystemRights -like "*read*"))
|
|
{
|
|
$_
|
|
}
|
|
}
|
|
$UserReadAclCount = $UserReadAcl | measure | select count -ExpandProperty count
|
|
|
|
# Get write counts
|
|
$UserWriteAcl = $UserAcls |
|
|
Foreach {
|
|
|
|
if(($_.FileSystemRights -like "*GenericAll*") -or ($_.FileSystemRights -like "*Write*"))
|
|
{
|
|
$_
|
|
}
|
|
}
|
|
$UserWriteAclCount = $UserWriteAcl | measure | select count -ExpandProperty count
|
|
|
|
# Get high risk counts
|
|
$UserHighRiskAcl = $UserAcls |
|
|
Foreach {
|
|
|
|
if(($_.ShareName -like 'c$') -or ($_.ShareName -like 'admin$') -or ($_.ShareName -like "*wwwroot*") -or ($_.ShareName -like "*inetpub*") -or ($_.ShareName -like 'c') -or ($_.ShareName -like 'c_share'))
|
|
{
|
|
$_
|
|
}
|
|
}
|
|
$UserHighRiskAclCount = $UserHighRiskAcl | measure | select count -ExpandProperty count
|
|
|
|
# Return object with all counts
|
|
$TheCounts = new-object psobject
|
|
$TheCounts | add-member Noteproperty UserAclsCount $UserAclsCount
|
|
$TheCounts | add-member Noteproperty UserShareCount $UserShareCount
|
|
$TheCounts | add-member Noteproperty UserComputerCount $UserComputerCount
|
|
$TheCounts | add-member Noteproperty UserReadAclCount $UserReadAclCount
|
|
$TheCounts | add-member Noteproperty UserWriteAclCount $UserWriteAclCount
|
|
$TheCounts | add-member Noteproperty UserHighRiskAclCount $UserHighRiskAclCount
|
|
$TheCounts
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Convert-DataTableToHtmlTable
|
|
# -------------------------------------------
|
|
function Convert-DataTableToHtmlTable
|
|
{
|
|
<#
|
|
.SYNOPSIS
|
|
This function can be used to convert a data table or ps object into a html table.
|
|
.PARAMETER $DataTable
|
|
The datatable to input.
|
|
.PARAMETER $Outfile
|
|
The output file path.
|
|
.PARAMETER $Title
|
|
Title of the page.
|
|
.PARAMETER $Description
|
|
Description of the page.
|
|
.EXAMPLE
|
|
$object = New-Object psobject
|
|
$Object | Add-Member Noteproperty Name "my name 1"
|
|
$Object | Add-Member Noteproperty Description "my description 1"
|
|
Convert-DataTableToHtmlTable -Verbose -DataTable $object -Outfile ".\MyPsDataTable.html" -Title "MyPage" -Description "My description" -DontExport
|
|
Convert-DataTableToHtmlTable -Verbose -DataTable $object -Outfile ".\MyPsDataTable.html" -Title "MyPage" -Description "My description"
|
|
.\MyPsDataTable.html
|
|
.NOTES
|
|
Author: Scott Sutherland (@_nullbind)
|
|
#>
|
|
param (
|
|
[Parameter(Mandatory = $true,
|
|
HelpMessage = 'The datatable to input.')]
|
|
$DataTable,
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'The output file path.')]
|
|
[string]$Outfile = ".\MyPsDataTable.html",
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Title of page.')]
|
|
[string]$Title = "HTML Table",
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Description of page.')]
|
|
[string]$Description = "HTML Table",
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Disable file save.')]
|
|
[switch]$DontExport
|
|
)
|
|
|
|
# Setup HTML begin
|
|
Write-Verbose "[+] Creating html top."
|
|
$HTMLSTART = @"
|
|
<html>
|
|
<head>
|
|
<title>$Title</title>
|
|
<style>
|
|
|
|
{box-sizing:border-box}
|
|
body,html{
|
|
font-family:"Open Sans",
|
|
sans-serif;font-weight:400;
|
|
min-height:100%;;color:#3d3935;
|
|
margin:1px;line-height:1.5;
|
|
}
|
|
|
|
table{
|
|
width:100%;
|
|
max-width:100%;
|
|
margin-bottom:1rem;
|
|
border-collapse:collapse
|
|
}
|
|
|
|
table thead th{
|
|
vertical-align:bottom;
|
|
border-bottom:2px solid #eceeef
|
|
}
|
|
|
|
table tbody tr:nth-of-type(odd){
|
|
background-color:#f9f9f9
|
|
}
|
|
|
|
table tbody tr:hover{
|
|
background-color:#f5f5f5
|
|
}
|
|
|
|
table td,table th{
|
|
padding:.75rem;
|
|
line-height:1.5;
|
|
text-align:left;
|
|
font-size:1rem;
|
|
vertical-align:top;
|
|
border-top:1px solid #eceeef
|
|
}
|
|
|
|
|
|
h1,h2,h3,h4,h5,h6{
|
|
padding-left: 10px;
|
|
font-family:inherit;
|
|
font-weight:500;
|
|
line-height:1.1;
|
|
color:inherit
|
|
}
|
|
|
|
code{
|
|
padding:.2rem .4rem;
|
|
font-size:1rem;
|
|
color:#bd4147;
|
|
background-color:#f7f7f9;
|
|
border-radius:.25rem
|
|
}
|
|
|
|
p{
|
|
margin-top:0;
|
|
margin-bottom:1rem
|
|
}
|
|
|
|
a,a:visited{
|
|
text-decoration:none;
|
|
font-size: 14;
|
|
color: gray;
|
|
font-weight: bold;
|
|
}
|
|
|
|
a:hover{
|
|
color:#9B3722;
|
|
text-decoration:underline
|
|
}
|
|
|
|
.link:hover{
|
|
text-decoration:underline
|
|
}
|
|
|
|
li{
|
|
list-style-type:none
|
|
}
|
|
|
|
.pageDescription {
|
|
margin: 10px;
|
|
}
|
|
|
|
.divbarDomain{
|
|
background:#d9d7d7;
|
|
width:200px;
|
|
border: 1px solid #999999;
|
|
height: 25px;
|
|
text-align:center;
|
|
}
|
|
|
|
.divbarDomainInside{
|
|
background:#9B3722;
|
|
width:100px;
|
|
text-align:center;
|
|
height: 25px;
|
|
vertical-align:middle;
|
|
}
|
|
.pageDescription {
|
|
padding-left: 0px;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<p class="pageDescription"><a href="javascript:history.back()">Back to Report</a></p>
|
|
<p class="pageDescription"><h3>$Title</h3></p>
|
|
<p class="pageDescription">$Description</p>
|
|
<table class="table table-striped table-hover">
|
|
"@
|
|
|
|
# Get list of columns
|
|
Write-Verbose "[+] Parsing data table columns."
|
|
$MyCsvColumns = $DataTable | Get-Member | Where-Object MemberType -like "NoteProperty" | Select-Object Name -ExpandProperty Name
|
|
|
|
# Print columns creation
|
|
Write-Verbose "[+] Creating html table columns."
|
|
$HTMLTableHeadStart= "<thead><tr>"
|
|
$MyCsvColumns |
|
|
ForEach-Object {
|
|
|
|
# Add column
|
|
$HTMLTableColumn = "<th>$_</th>$HTMLTableColumn"
|
|
}
|
|
$HTMLTableColumn = "$HTMLTableHeadStart$HTMLTableColumn</tr></thead>"
|
|
|
|
# Create table rows
|
|
Write-Verbose "[+] Creating html table rows."
|
|
$HTMLTableRow = $DataTable |
|
|
ForEach-Object {
|
|
|
|
# Create a value contain row data
|
|
$CurrentRow = $_
|
|
$PrintRow = ""
|
|
$MyCsvColumns |
|
|
ForEach-Object{
|
|
|
|
try{
|
|
$GetValue = $CurrentRow | Select-Object $_ -ExpandProperty $_ -ErrorAction SilentlyContinue
|
|
if($PrintRow -eq ""){
|
|
$PrintRow = "<td>$GetValue</td>"
|
|
}else{
|
|
$PrintRow = "<td>$GetValue</td>$PrintRow"
|
|
}
|
|
}catch{}
|
|
}
|
|
|
|
# Return row
|
|
$HTMLTableHeadstart = "<tr>"
|
|
$HTMLTableHeadend = "</tr>"
|
|
"$HTMLTableHeadStart$PrintRow$HTMLTableHeadend"
|
|
}
|
|
|
|
# Setup HTML end
|
|
Write-Verbose "[+] Creating html bottom."
|
|
$HTMLEND = @"
|
|
</tbody>
|
|
</table>
|
|
</body>
|
|
</html>
|
|
"@
|
|
|
|
# Return it
|
|
"$HTMLSTART $HTMLTableColumn $HTMLTableRow $HTMLEND"
|
|
|
|
# Write file
|
|
if(-not $DontExport){
|
|
"$HTMLSTART $HTMLTableColumn $HTMLTableRow $HTMLEND" | Out-File $Outfile
|
|
}
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Convert-DataTableToHtmlReport
|
|
# -------------------------------------------
|
|
function Convert-DataTableToHtmlReport
|
|
{
|
|
<#
|
|
.SYNOPSIS
|
|
This function can be used to convert a data table or ps object into a generic html table.
|
|
.PARAMETER $DataTable
|
|
The datatable to input.
|
|
.EXAMPLE
|
|
$object = New-Object psobject
|
|
$Object | Add-Member Noteproperty Name "my name 1"
|
|
$Object | Add-Member Noteproperty Description "my description 1"
|
|
Convert-DataTableToHtmlReport -Verbose -DataTable $object
|
|
.NOTES
|
|
Author: Scott Sutherland (@_nullbind)
|
|
#>
|
|
param (
|
|
[Parameter(Mandatory = $true,
|
|
HelpMessage = 'The datatable to input.')]
|
|
$DataTable
|
|
)
|
|
|
|
# Setup HTML begin
|
|
Write-Verbose "[+] Creating html top."
|
|
$HTMLSTART = @"
|
|
<table class="table table-striped table-hover tabledrop">
|
|
"@
|
|
|
|
# Get list of columns
|
|
$MyCsvColumns = $DataTable | Get-Member | Where-Object MemberType -like "NoteProperty" | Select-Object Name -ExpandProperty Name
|
|
|
|
# Print columns creation
|
|
$HTMLTableHeadStart= "<thead><tr>"
|
|
$MyCsvColumns |
|
|
ForEach-Object {
|
|
|
|
# Add column
|
|
$HTMLTableColumn = "<th>$_</th>$HTMLTableColumn"
|
|
}
|
|
$HTMLTableColumn = "$HTMLTableHeadStart$HTMLTableColumn</tr></thead>"
|
|
|
|
# Create table rows
|
|
Write-Verbose "[+] Creating html table rows."
|
|
$HTMLTableRow = $DataTable |
|
|
ForEach-Object {
|
|
|
|
# Create a value contain row data
|
|
$CurrentRow = $_
|
|
$PrintRow = ""
|
|
$MyCsvColumns |
|
|
ForEach-Object{
|
|
|
|
try{
|
|
$GetValue = $CurrentRow | Select-Object $_ -ExpandProperty $_ -ErrorAction SilentlyContinue
|
|
if($PrintRow -eq ""){
|
|
$PrintRow = "<td>$GetValue</td>"
|
|
}else{
|
|
$PrintRow = "<td>$GetValue</td>$PrintRow"
|
|
}
|
|
}catch{}
|
|
}
|
|
|
|
# Return row
|
|
$HTMLTableHeadstart = "<tr>"
|
|
$HTMLTableHeadend = "</tr>"
|
|
"$HTMLTableHeadStart$PrintRow$HTMLTableHeadend"
|
|
}
|
|
|
|
# Setup HTML end
|
|
Write-Verbose "[+] Creating html bottom."
|
|
$HTMLEND = @"
|
|
</tbody>
|
|
</table>
|
|
"@
|
|
|
|
# Return it
|
|
"$HTMLSTART $HTMLTableColumn $HTMLTableRow $HTMLEND"
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-FolderGroupMd5
|
|
# -------------------------------------------
|
|
function Get-FolderGroupMd5{
|
|
|
|
param (
|
|
[string]$FolderList
|
|
)
|
|
|
|
<#
|
|
$stringAsStream = [System.IO.MemoryStream]::new()
|
|
$writer = [System.IO.StreamWriter]::new($stringAsStream)
|
|
$writer.write($FolderList)
|
|
$writer.Flush()
|
|
$stringAsStream.Position = 0
|
|
Get-FileHash -InputStream $stringAsStream -Algorithm MD5 | Select-Object Hash
|
|
#>
|
|
|
|
$MyMd5Provider = [System.Security.Cryptography.MD5CryptoServiceProvider]::Create()
|
|
$enc = [system.Text.Encoding]::UTF8
|
|
$FolderListBytes = $enc.GetBytes($FolderList)
|
|
$MyMd5HashBytes = $MyMd5Provider.ComputeHash($FolderListBytes)
|
|
$MysStringBuilder = new-object System.Text.StringBuilder
|
|
$MyMd5HashBytes|
|
|
foreach {
|
|
$MyMd5HashByte = $_.ToString("x2").ToLower()
|
|
$MyMd5Hash = "$MyMd5Hash$MyMd5HashByte"
|
|
}
|
|
$MyMd5Hash
|
|
}
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-LdapQuery
|
|
# -------------------------------------------
|
|
# Author: Will Schroeder
|
|
# Modifications: Scott Sutherland
|
|
function Get-LdapQuery
|
|
{
|
|
<#
|
|
.SYNOPSIS
|
|
Used to query domain controllers via LDAP. Supports alternative credentials from non-domain system
|
|
Note: This will use the default logon server by default.
|
|
.PARAMETER Username
|
|
Domain account to authenticate to Active Directory.
|
|
.PARAMETER Password
|
|
Domain password to authenticate to Active Directory.
|
|
.PARAMETER Credential
|
|
Domain credential to authenticate to Active Directory.
|
|
.PARAMETER DomainController
|
|
Domain controller to authenticated to. Requires username/password or credential.
|
|
.PARAMETER LdapFilter
|
|
LDAP filter.
|
|
.PARAMETER LdapPath
|
|
Ldap path.
|
|
.PARAMETER $Limit
|
|
Maximum number of Objects to pull from AD, limit is 1,000.".
|
|
.PARAMETER SearchScope
|
|
Scope of a search as either a base, one-level, or subtree search, default is subtree..
|
|
.EXAMPLE
|
|
PS C:\temp> Get-DomainObject -LdapFilter "(&(servicePrincipalName=*))"
|
|
.EXAMPLE
|
|
PS C:\temp> Get-DomainObject -LdapFilter "(&(servicePrincipalName=*))" -DomainController 10.0.0.1:389
|
|
It will use the security context of the current process to authenticate to the domain controller.
|
|
IP:Port can be specified to reach a pivot machine.
|
|
.EXAMPLE
|
|
PS C:\temp> Get-DomainObject -LdapFilter "(&(servicePrincipalName=*))" -DomainController 10.0.0.1 -Username Domain\User -Password Password123!
|
|
.Notes
|
|
This was based on Will Schroeder's Get-ADObject function from https://github.com/PowerShellEmpire/PowerTools/blob/master/PowerView/powerview.ps1
|
|
#>
|
|
[CmdletBinding()]
|
|
Param(
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Domain user to authenticate with domain\user.')]
|
|
[string]$Username,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Domain password to authenticate with domain\user.')]
|
|
[string]$Password,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Credentials to use when connecting to a Domain Controller.')]
|
|
[System.Management.Automation.PSCredential]
|
|
[System.Management.Automation.Credential()]$Credential = [System.Management.Automation.PSCredential]::Empty,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Domain controller for Domain and Site that you want to query against.')]
|
|
[string]$DomainController,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'LDAP Filter.')]
|
|
[string]$LdapFilter = '',
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'LDAP path.')]
|
|
[string]$LdapPath,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'Maximum number of Objects to pull from AD, limit is 1,000 .')]
|
|
[int]$Limit = 1000,
|
|
|
|
[Parameter(Mandatory = $false,
|
|
HelpMessage = 'scope of a search as either a base, one-level, or subtree search, default is subtree.')]
|
|
[ValidateSet('Subtree','OneLevel','Base')]
|
|
[string]$SearchScope = 'Subtree'
|
|
)
|
|
Begin
|
|
{
|
|
# Create PS Credential object
|
|
if($Username -and $Password)
|
|
{
|
|
$secpass = ConvertTo-SecureString $Password -AsPlainText -Force
|
|
$Credential = New-Object -TypeName System.Management.Automation.PSCredential -ArgumentList ($Username, $secpass)
|
|
}
|
|
|
|
# Create Create the connection to LDAP
|
|
if ($DomainController)
|
|
{
|
|
|
|
# Test credentials and grab domain
|
|
try {
|
|
|
|
$ArgumentList = New-Object Collections.Generic.List[string]
|
|
$ArgumentList.Add("LDAP://$DomainController")
|
|
|
|
if($Username -or $Credential){
|
|
$ArgumentList.Add($Credential.UserName)
|
|
$ArgumentList.Add($Credential.GetNetworkCredential().Password)
|
|
}
|
|
|
|
$objDomain = (New-Object -TypeName System.DirectoryServices.DirectoryEntry -ArgumentList $ArgumentList).distinguishedname
|
|
|
|
# Authentication failed. distinguishedName property can not be empty.
|
|
if(-not $objDomain){ throw }
|
|
|
|
}catch{
|
|
$Time = Get-Date -UFormat "%m/%d/%Y %R"
|
|
Write-Host " [!][$Time] Authentication failed or domain controller is not reachable."
|
|
Write-Host " [!][$Time] Aborting operation."
|
|
Break
|
|
}
|
|
|
|
# add ldap path
|
|
if($LdapPath)
|
|
{
|
|
$LdapPath = '/'+$LdapPath+','+$objDomain
|
|
$ArgumentList[0] = "LDAP://$DomainController$LdapPath"
|
|
}
|
|
|
|
$objDomainPath= New-Object System.DirectoryServices.DirectoryEntry -ArgumentList $ArgumentList
|
|
|
|
$objSearcher = New-Object -TypeName System.DirectoryServices.DirectorySearcher $objDomainPath
|
|
}
|
|
else
|
|
{
|
|
$objDomain = ([ADSI]'').distinguishedName
|
|
|
|
# add ldap path
|
|
if($LdapPath)
|
|
{
|
|
$LdapPath = $LdapPath+','+$objDomain
|
|
$objDomainPath = [ADSI]"LDAP://$LdapPath"
|
|
}
|
|
else
|
|
{
|
|
$objDomainPath = [ADSI]''
|
|
}
|
|
|
|
$objSearcher = New-Object -TypeName System.DirectoryServices.DirectorySearcher -ArgumentList $objDomainPath
|
|
}
|
|
|
|
# Setup LDAP filter
|
|
$objSearcher.PageSize = $Limit
|
|
$objSearcher.Filter = $LdapFilter
|
|
$objSearcher.SearchScope = 'Subtree'
|
|
}
|
|
|
|
Process
|
|
{
|
|
try
|
|
{
|
|
# Return object
|
|
$objSearcher.FindAll() | ForEach-Object -Process {
|
|
$_
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
"Error was $_"
|
|
$line = $_.InvocationInfo.ScriptLineNumber
|
|
"Error was in Line $line"
|
|
}
|
|
}
|
|
|
|
End
|
|
{
|
|
}
|
|
}
|
|
|
|
|
|
# -------------------------------------------
|
|
# Function: Get-DomainSubnet
|
|
# -------------------------------------------
|
|
function Get-DomainSubnet
|
|
{
|
|
[CmdletBinding()]
|
|
Param(
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="Domain user to authenticate with domain\user.")]
|
|
[string]$username,
|
|
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="Domain password to authenticate with domain\user.")]
|
|
[string]$password,
|
|
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="Credentials to use when connecting to a Domain Controller.")]
|
|
[System.Management.Automation.PSCredential]
|
|
[System.Management.Automation.Credential()]$Credential = [System.Management.Automation.PSCredential]::Empty,
|
|
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="Domain controller for Domain and Site that you want to query against.")]
|
|
[string]$DomainController,
|
|
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="LDAP Filter.")]
|
|
[string]$LdapFilter = "",
|
|
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="LDAP path.")]
|
|
[string]$LdapPath,
|
|
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="Maximum number of Objects to pull from AD, limit is 1,000 .")]
|
|
[int]$Limit = 1000,
|
|
|
|
[Parameter(Mandatory=$false,
|
|
HelpMessage="scope of a search as either a base, one-level, or subtree search, default is subtree.")]
|
|
[ValidateSet("Subtree","OneLevel","Base")]
|
|
[string]$SearchScope = "Subtree"
|
|
)
|
|
Begin
|
|
{
|
|
Write-Verbose "Getting subnets..."
|
|
|
|
# Create PS Credential object
|
|
if($Password){
|
|
$secpass = ConvertTo-SecureString $Password -AsPlainText -Force
|
|
$Credential = New-Object System.Management.Automation.PSCredential ($Username, $secpass)
|
|
}
|
|
|
|
# Create Create the connection to LDAP
|
|
if ($DomainController -and $Credential.GetNetworkCredential().Password)
|
|
{
|
|
$objDomain = (New-Object System.DirectoryServices.DirectoryEntry "LDAP://$DomainController", $Credential.UserName,$Credential.GetNetworkCredential().Password).distinguishedname
|
|
|
|
# add ldap path
|
|
if($LdapPath)
|
|
{
|
|
$LdapPath = "/"+$LdapPath+","+$objDomain
|
|
$objDomainPath = New-Object System.DirectoryServices.DirectoryEntry "LDAP://$DomainController$LdapPath", $Credential.UserName,$Credential.GetNetworkCredential().Password
|
|
}else{
|
|
$objDomainPath= New-Object System.DirectoryServices.DirectoryEntry "LDAP://$DomainController", $Credential.UserName,$Credential.GetNetworkCredential().Password
|
|
}
|
|
|
|
$objSearcher = New-Object System.DirectoryServices.DirectorySearcher $objDomainPath
|
|
}else{
|
|
$objDomain = ([ADSI]"").distinguishedName
|
|
|
|
# add ldap path
|
|
if($LdapPath)
|
|
{
|
|
$LdapPath = $LdapPath+","+$objDomain
|
|
$LdapPath
|
|
$objDomainPath = [ADSI]"LDAP://$LdapPath"
|
|
}else{
|
|
$objDomainPath = [ADSI]""
|
|
}
|
|
|
|
$objSearcher = New-Object System.DirectoryServices.DirectorySearcher $objDomainPath
|
|
}
|
|
|
|
# Create table for object information
|
|
$TableDomainSubnets = New-Object System.Data.DataTable
|
|
$TableDomainSubnets.Columns.Add("Site") | Out-Null
|
|
$TableDomainSubnets.Columns.Add("Subnet") | Out-Null
|
|
$TableDomainSubnets.Columns.Add("Description") | Out-Null
|
|
$TableDomainSubnets.Columns.Add("whencreated") | Out-Null
|
|
$TableDomainSubnets.Columns.Add("whenchanged") | Out-Null
|
|
$TableDomainSubnets.Columns.Add("distinguishedname") | Out-Null
|
|
$TableDomainSubnets.Clear()
|
|
}
|
|
|
|
Process
|
|
{
|
|
try
|
|
{
|
|
# Get results
|
|
Get-LdapQuery -DomainController $DomainController -username $username -password $password -Credential $Credential -Verbose -LdapFilter "(objectCategory=subnet)" -LdapPath "CN=Subnets,CN=Sites,CN=Configuration" |
|
|
ForEach-Object {
|
|
|
|
# Add results to table
|
|
$TableDomainSubnets.Rows.Add(
|
|
[string]$_.properties.siteobject.split(",")[0].split("=")[1],
|
|
[string]$_.properties.name,
|
|
[string]$_.properties.description,
|
|
[string]$_.properties.whencreated,
|
|
[string]$_.properties.whenchanged,
|
|
[string]$_.properties.distinguishedname
|
|
) | Out-Null
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
#"Error was $_"
|
|
#$line = $_.InvocationInfo.ScriptLineNumber
|
|
#"Error was in Line $line"
|
|
}
|
|
}
|
|
|
|
End
|
|
{
|
|
# Check for subnets
|
|
if($TableDomainSubnets.Rows.Count -gt 0)
|
|
{
|
|
$TableDomainSubnetsCount = $TableDomainSubnets.Rows.Count
|
|
Write-Verbose "$TableDomainSubnetsCount subnets found."
|
|
Return $TableDomainSubnets
|
|
}else{
|
|
Write-Verbose "0 subnets were found."
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
# -------------------------------------------
|
|
# Function: Invoke-Parallel
|
|
# -------------------------------------------
|
|
# Author: RamblingCookieMonster
|
|
# Source: https://github.com/RamblingCookieMonster/Invoke-Parallel
|
|
# Notes: Added "ImportSessionFunctions" to import custom functions from the current session into the runspace pool.
|
|
function Invoke-Parallel
|
|
{
|
|
<#
|
|
.SYNOPSIS
|
|
Function to control parallel processing using runspaces
|
|
|
|
.DESCRIPTION
|
|
Function to control parallel processing using runspaces
|
|
|
|
Note that each runspace will not have access to variables and commands loaded in your session or in other runspaces by default.
|
|
This behaviour can be changed with parameters.
|
|
|
|
.PARAMETER ScriptFile
|
|
File to run against all input objects. Must include parameter to take in the input object, or use $args. Optionally, include parameter to take in parameter. Example: C:\script.ps1
|
|
|
|
.PARAMETER ScriptBlock
|
|
Scriptblock to run against all computers.
|
|
|
|
You may use $Using:<Variable> language in PowerShell 3 and later.
|
|
|
|
The parameter block is added for you, allowing behaviour similar to foreach-object:
|
|
Refer to the input object as $_.
|
|
Refer to the parameter parameter as $parameter
|
|
|
|
.PARAMETER InputObject
|
|
Run script against these specified objects.
|
|
|
|
.PARAMETER Parameter
|
|
This object is passed to every script block. You can use it to pass information to the script block; for example, the path to a logging folder
|
|
|
|
Reference this object as $parameter if using the scriptblock parameterset.
|
|
|
|
.PARAMETER ImportVariables
|
|
If specified, get user session variables and add them to the initial session state
|
|
|
|
.PARAMETER ImportModules
|
|
If specified, get loaded modules and pssnapins, add them to the initial session state
|
|
|
|
.PARAMETER Throttle
|
|
Maximum number of threads to run at a single time.
|
|
|
|
.PARAMETER SleepTimer
|
|
Milliseconds to sleep after checking for completed runspaces and in a few other spots. I would not recommend dropping below 200 or increasing above 500
|
|
|
|
.PARAMETER RunspaceTimeout
|
|
Maximum time in seconds a single thread can run. If execution of your code takes longer than this, it is disposed. Default: 0 (seconds)
|
|
|
|
WARNING: Using this parameter requires that maxQueue be set to throttle (it will be by default) for accurate timing. Details here:
|
|
http://gallery.technet.microsoft.com/Run-Parallel-Parallel-377fd430
|
|
|
|
.PARAMETER NoCloseOnTimeout
|
|
Do not dispose of timed out tasks or attempt to close the runspace if threads have timed out. This will prevent the script from hanging in certain situations where threads become non-responsive, at the expense of leaking memory within the PowerShell host.
|
|
|
|
.PARAMETER MaxQueue
|
|
Maximum number of powershell instances to add to runspace pool. If this is higher than $throttle, $timeout will be inaccurate
|
|
|
|
If this is equal or less than throttle, there will be a performance impact
|
|
|
|
The default value is $throttle times 3, if $runspaceTimeout is not specified
|
|
The default value is $throttle, if $runspaceTimeout is specified
|
|
|
|
.PARAMETER LogFile
|
|
Path to a file where we can log results, including run time for each thread, whether it completes, completes with errors, or times out.
|
|
|
|
.PARAMETER Quiet
|
|
Disable progress bar.
|
|
|
|
.EXAMPLE
|
|
Each example uses Test-ForPacs.ps1 which includes the following code:
|
|
param($computer)
|
|
|
|
if(test-connection $computer -count 1 -quiet -BufferSize 16){
|
|
$object = [pscustomobject] @{
|
|
Computer=$computer;
|
|
Available=1;
|
|
Kodak=$(
|
|
if((test-path "\\$computer\c$\users\public\desktop\Kodak Direct View Pacs.url") -or (test-path "\\$computer\c$\documents and settings\all users
|
|
|
|
\desktop\Kodak Direct View Pacs.url") ){"1"}else{"0"}
|
|
)
|
|
}
|
|
}
|
|
else{
|
|
$object = [pscustomobject] @{
|
|
Computer=$computer;
|
|
Available=0;
|
|
Kodak="NA"
|
|
}
|
|
}
|
|
|
|
$object
|
|
|
|
.EXAMPLE
|
|
Invoke-Parallel -scriptfile C:\public\Test-ForPacs.ps1 -inputobject $(get-content C:\pcs.txt) -runspaceTimeout 10 -throttle 10
|
|
|
|
Pulls list of PCs from C:\pcs.txt,
|
|
Runs Test-ForPacs against each
|
|
If any query takes longer than 10 seconds, it is disposed
|
|
Only run 10 threads at a time
|
|
|
|
.EXAMPLE
|
|
Invoke-Parallel -scriptfile C:\public\Test-ForPacs.ps1 -inputobject c-is-ts-91, c-is-ts-95
|
|
|
|
Runs against c-is-ts-91, c-is-ts-95 (-computername)
|
|
Runs Test-ForPacs against each
|
|
|
|
.EXAMPLE
|
|
$stuff = [pscustomobject] @{
|
|
ContentFile = "windows\system32\drivers\etc\hosts"
|
|
Logfile = "C:\temp\log.txt"
|
|
}
|
|
|
|
$computers | Invoke-Parallel -parameter $stuff {
|
|
$contentFile = join-path "\\$_\c$" $parameter.contentfile
|
|
Get-Content $contentFile |
|
|
set-content $parameter.logfile
|
|
}
|
|
|
|
This example uses the parameter argument. This parameter is a single object. To pass multiple items into the script block, we create a custom object (using a PowerShell v3 language) with properties we want to pass in.
|
|
|
|
Inside the script block, $parameter is used to reference this parameter object. This example sets a content file, gets content from that file, and sets it to a predefined log file.
|
|
|
|
.EXAMPLE
|
|
$test = 5
|
|
1..2 | Invoke-Parallel -ImportVariables {$_ * $test}
|
|
|
|
Add variables from the current session to the session state. Without -ImportVariables $Test would not be accessible
|
|
|
|
.EXAMPLE
|
|
$test = 5
|
|
1..2 | Invoke-Parallel {$_ * $Using:test}
|
|
|
|
Reference a variable from the current session with the $Using:<Variable> syntax. Requires PowerShell 3 or later. Note that -ImportVariables parameter is no longer necessary.
|
|
|
|
.FUNCTIONALITY
|
|
PowerShell Language
|
|
|
|
.NOTES
|
|
Credit to Boe Prox for the base runspace code and $Using implementation
|
|
http://learn-powershell.net/2012/05/10/speedy-network-information-query-using-powershell/
|
|
http://gallery.technet.microsoft.com/scriptcenter/Speedy-Network-Information-5b1406fb#content
|
|
https://github.com/proxb/PoshRSJob/
|
|
|
|
Credit to T Bryce Yehl for the Quiet and NoCloseOnTimeout implementations
|
|
|
|
Credit to Sergei Vorobev for the many ideas and contributions that have improved functionality, reliability, and ease of use
|
|
|
|
.LINK
|
|
https://github.com/RamblingCookieMonster/Invoke-Parallel
|
|
#>
|
|
[cmdletbinding(DefaultParameterSetName = 'ScriptBlock')]
|
|
Param (
|
|
[Parameter(Mandatory = $false,position = 0,ParameterSetName = 'ScriptBlock')]
|
|
[System.Management.Automation.ScriptBlock]$ScriptBlock,
|
|
|
|
[Parameter(Mandatory = $false,ParameterSetName = 'ScriptFile')]
|
|
[ValidateScript({
|
|
Test-Path $_ -PathType leaf
|
|
})]
|
|
$ScriptFile,
|
|
|
|
[Parameter(Mandatory = $true,ValueFromPipeline = $true)]
|
|
[Alias('CN','__Server','IPAddress','Server','ComputerName')]
|
|
[PSObject]$InputObject,
|
|
|
|
[PSObject]$Parameter,
|
|
|
|
[switch]$ImportSessionFunctions,
|
|
|
|
[switch]$ImportVariables,
|
|
|
|
[switch]$ImportModules,
|
|
|
|
[int]$Throttle = 20,
|
|
|
|
[int]$SleepTimer = 200,
|
|
|
|
[int]$RunspaceTimeout = 0,
|
|
|
|
[switch]$NoCloseOnTimeout = $false,
|
|
|
|
[int]$MaxQueue,
|
|
|
|
[validatescript({
|
|
Test-Path (Split-Path -Path $_ -Parent)
|
|
})]
|
|
[string]$LogFile = 'C:\temp\log.log',
|
|
|
|
[switch] $Quiet = $true
|
|
)
|
|
|
|
Begin {
|
|
|
|
#No max queue specified? Estimate one.
|
|
#We use the script scope to resolve an odd PowerShell 2 issue where MaxQueue isn't seen later in the function
|
|
if( -not $PSBoundParameters.ContainsKey('MaxQueue') )
|
|
{
|
|
if($RunspaceTimeout -ne 0)
|
|
{
|
|
$script:MaxQueue = $Throttle
|
|
}
|
|
else
|
|
{
|
|
$script:MaxQueue = $Throttle * 3
|
|
}
|
|
}
|
|
else
|
|
{
|
|
$script:MaxQueue = $MaxQueue
|
|
}
|
|
|
|
#If they want to import variables or modules, create a clean runspace, get loaded items, use those to exclude items
|
|
if ($ImportVariables -or $ImportModules)
|
|
{
|
|
$StandardUserEnv = [powershell]::Create().addscript({
|
|
#Get modules and snapins in this clean runspace
|
|
$Modules = Get-Module | Select-Object -ExpandProperty Name
|
|
$Snapins = Get-PSSnapin | Select-Object -ExpandProperty Name
|
|
|
|
#Get variables in this clean runspace
|
|
#Called last to get vars like $? into session
|
|
$Variables = Get-Variable | Select-Object -ExpandProperty Name
|
|
|
|
#Return a hashtable where we can access each.
|
|
@{
|
|
Variables = $Variables
|
|
Modules = $Modules
|
|
Snapins = $Snapins
|
|
}
|
|
}).invoke()[0]
|
|
|
|
if ($ImportVariables)
|
|
{
|
|
#Exclude common parameters, bound parameters, and automatic variables
|
|
Function _temp
|
|
{
|
|
[cmdletbinding()] param()
|
|
}
|
|
$VariablesToExclude = @( (Get-Command _temp | Select-Object -ExpandProperty parameters).Keys + $PSBoundParameters.Keys + $StandardUserEnv.Variables )
|
|
#Write-Verbose "Excluding variables $( ($VariablesToExclude | sort ) -join ", ")"
|
|
|
|
# we don't use 'Get-Variable -Exclude', because it uses regexps.
|
|
# One of the veriables that we pass is '$?'.
|
|
# There could be other variables with such problems.
|
|
# Scope 2 required if we move to a real module
|
|
$UserVariables = @( Get-Variable | Where-Object -FilterScript {
|
|
-not ($VariablesToExclude -contains $_.Name)
|
|
} )
|
|
#Write-Verbose "Found variables to import: $( ($UserVariables | Select -expandproperty Name | Sort ) -join ", " | Out-String).`n"
|
|
}
|
|
|
|
if ($ImportModules)
|
|
{
|
|
$UserModules = @( Get-Module |
|
|
Where-Object -FilterScript {
|
|
$StandardUserEnv.Modules -notcontains $_.Name -and (Test-Path -Path $_.Path -ErrorAction SilentlyContinue)
|
|
} |
|
|
Select-Object -ExpandProperty Path )
|
|
$UserSnapins = @( Get-PSSnapin |
|
|
Select-Object -ExpandProperty Name |
|
|
Where-Object -FilterScript {
|
|
$StandardUserEnv.Snapins -notcontains $_
|
|
} )
|
|
}
|
|
}
|
|
|
|
#region functions
|
|
|
|
Function Get-RunspaceData
|
|
{
|
|
[cmdletbinding()]
|
|
param( [switch]$Wait )
|
|
|
|
#loop through runspaces
|
|
#if $wait is specified, keep looping until all complete
|
|
Do
|
|
{
|
|
#set more to false for tracking completion
|
|
$more = $false
|
|
|
|
#Progress bar if we have inputobject count (bound parameter)
|
|
if (-not $Quiet)
|
|
{
|
|
Write-Progress -Activity 'Running Query' -Status 'Starting threads'`
|
|
-CurrentOperation "$startedCount threads defined - $totalCount input objects - $script:completedCount input objects processed"`
|
|
-PercentComplete $( Try
|
|
{
|
|
$script:completedCount / $totalCount * 100
|
|
}
|
|
Catch
|
|
{
|
|
0
|
|
}
|
|
)
|
|
}
|
|
|
|
#run through each runspace.
|
|
Foreach($runspace in $runspaces)
|
|
{
|
|
#get the duration - inaccurate
|
|
$currentdate = Get-Date
|
|
$runtime = $currentdate - $runspace.startTime
|
|
$runMin = [math]::Round( $runtime.totalminutes ,2 )
|
|
|
|
#set up log object
|
|
$log = '' | Select-Object -Property Date, Action, Runtime, Status, Details
|
|
$log.Action = "Removing:'$($runspace.object)'"
|
|
$log.Date = $currentdate
|
|
$log.Runtime = "$runMin minutes"
|
|
|
|
#If runspace completed, end invoke, dispose, recycle, counter++
|
|
If ($runspace.Runspace.isCompleted)
|
|
{
|
|
$script:completedCount++
|
|
|
|
#check if there were errors
|
|
if($runspace.powershell.Streams.Error.Count -gt 0)
|
|
{
|
|
#set the logging info and move the file to completed
|
|
$log.status = 'CompletedWithErrors'
|
|
#Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
foreach($ErrorRecord in $runspace.powershell.Streams.Error)
|
|
{
|
|
Write-Error -ErrorRecord $ErrorRecord
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#add logging details and cleanup
|
|
$log.status = 'Completed'
|
|
#Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
}
|
|
|
|
#everything is logged, clean up the runspace
|
|
$runspace.powershell.EndInvoke($runspace.Runspace)
|
|
$runspace.powershell.dispose()
|
|
$runspace.Runspace = $null
|
|
$runspace.powershell = $null
|
|
}
|
|
|
|
#If runtime exceeds max, dispose the runspace
|
|
ElseIf ( $RunspaceTimeout -ne 0 -and $runtime.totalseconds -gt $RunspaceTimeout)
|
|
{
|
|
$script:completedCount++
|
|
$timedOutTasks = $true
|
|
|
|
#add logging details and cleanup
|
|
$log.status = 'TimedOut'
|
|
#Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
Write-Error -Message "Runspace timed out at $($runtime.totalseconds) seconds for the object:`n$($runspace.object | Out-String)"
|
|
|
|
#Depending on how it hangs, we could still get stuck here as dispose calls a synchronous method on the powershell instance
|
|
if (!$NoCloseOnTimeout)
|
|
{
|
|
$runspace.powershell.dispose()
|
|
}
|
|
$runspace.Runspace = $null
|
|
$runspace.powershell = $null
|
|
$completedCount++
|
|
}
|
|
|
|
#If runspace isn't null set more to true
|
|
ElseIf ($runspace.Runspace -ne $null )
|
|
{
|
|
$log = $null
|
|
$more = $true
|
|
}
|
|
}
|
|
|
|
#Clean out unused runspace jobs
|
|
$temphash = $runspaces.clone()
|
|
$temphash |
|
|
Where-Object -FilterScript {
|
|
$_.runspace -eq $null
|
|
} |
|
|
ForEach-Object -Process {
|
|
$runspaces.remove($_)
|
|
}
|
|
|
|
#sleep for a bit if we will loop again
|
|
if($PSBoundParameters['Wait'])
|
|
{
|
|
Start-Sleep -Milliseconds $SleepTimer
|
|
}
|
|
|
|
#Loop again only if -wait parameter and there are more runspaces to process
|
|
}
|
|
while ($more -and $PSBoundParameters['Wait'])
|
|
|
|
#End of runspace function
|
|
}
|
|
|
|
#endregion functions
|
|
|
|
#region Init
|
|
|
|
if($PSCmdlet.ParameterSetName -eq 'ScriptFile')
|
|
{
|
|
$ScriptBlock = [scriptblock]::Create( $(Get-Content $ScriptFile | Out-String) )
|
|
}
|
|
elseif($PSCmdlet.ParameterSetName -eq 'ScriptBlock')
|
|
{
|
|
#Start building parameter names for the param block
|
|
[string[]]$ParamsToAdd = '$_'
|
|
if( $PSBoundParameters.ContainsKey('Parameter') )
|
|
{
|
|
$ParamsToAdd += '$Parameter'
|
|
}
|
|
|
|
$UsingVariableData = $null
|
|
|
|
|
|
# This code enables $Using support through the AST.
|
|
# This is entirely from Boe Prox, and his https://github.com/proxb/PoshRSJob module; all credit to Boe!
|
|
|
|
if($PSVersionTable.PSVersion.Major -gt 2)
|
|
{
|
|
#Extract using references
|
|
$UsingVariables = $ScriptBlock.ast.FindAll({
|
|
$args[0] -is [System.Management.Automation.Language.UsingExpressionAst]
|
|
},$true)
|
|
|
|
If ($UsingVariables)
|
|
{
|
|
$List = New-Object -TypeName 'System.Collections.Generic.List`1[System.Management.Automation.Language.VariableExpressionAst]'
|
|
ForEach ($Ast in $UsingVariables)
|
|
{
|
|
[void]$List.Add($Ast.SubExpression)
|
|
}
|
|
|
|
$UsingVar = $UsingVariables |
|
|
Group-Object -Property SubExpression |
|
|
ForEach-Object -Process {
|
|
$_.Group |
|
|
Select-Object -First 1
|
|
}
|
|
|
|
#Extract the name, value, and create replacements for each
|
|
$UsingVariableData = ForEach ($Var in $UsingVar)
|
|
{
|
|
Try
|
|
{
|
|
$Value = Get-Variable -Name $Var.SubExpression.VariablePath.UserPath -ErrorAction Stop
|
|
[pscustomobject]@{
|
|
Name = $Var.SubExpression.Extent.Text
|
|
Value = $Value.Value
|
|
NewName = ('$__using_{0}' -f $Var.SubExpression.VariablePath.UserPath)
|
|
NewVarName = ('__using_{0}' -f $Var.SubExpression.VariablePath.UserPath)
|
|
}
|
|
}
|
|
Catch
|
|
{
|
|
Write-Error -Message "$($Var.SubExpression.Extent.Text) is not a valid Using: variable!"
|
|
}
|
|
}
|
|
$ParamsToAdd += $UsingVariableData | Select-Object -ExpandProperty NewName -Unique
|
|
|
|
$NewParams = $UsingVariableData.NewName -join ', '
|
|
$Tuple = [Tuple]::Create($List, $NewParams)
|
|
$bindingFlags = [Reflection.BindingFlags]'Default,NonPublic,Instance'
|
|
$GetWithInputHandlingForInvokeCommandImpl = ($ScriptBlock.ast.gettype().GetMethod('GetWithInputHandlingForInvokeCommandImpl',$bindingFlags))
|
|
|
|
$StringScriptBlock = $GetWithInputHandlingForInvokeCommandImpl.Invoke($ScriptBlock.ast,@($Tuple))
|
|
|
|
$ScriptBlock = [scriptblock]::Create($StringScriptBlock)
|
|
|
|
#Write-Verbose $StringScriptBlock
|
|
}
|
|
}
|
|
|
|
$ScriptBlock = $ExecutionContext.InvokeCommand.NewScriptBlock("param($($ParamsToAdd -Join ', '))`r`n" + $ScriptBlock.ToString())
|
|
}
|
|
else
|
|
{
|
|
Throw 'Must provide ScriptBlock or ScriptFile'
|
|
Break
|
|
}
|
|
|
|
Write-Debug -Message "`$ScriptBlock: $($ScriptBlock | Out-String)"
|
|
If (-not($SuppressVerbose)){
|
|
Write-Verbose -Message 'Creating runspace pool and session states'
|
|
}
|
|
|
|
|
|
#If specified, add variables and modules/snapins to session state
|
|
$sessionstate = [System.Management.Automation.Runspaces.InitialSessionState]::CreateDefault()
|
|
if ($ImportVariables)
|
|
{
|
|
if($UserVariables.count -gt 0)
|
|
{
|
|
foreach($Variable in $UserVariables)
|
|
{
|
|
$sessionstate.Variables.Add( (New-Object -TypeName System.Management.Automation.Runspaces.SessionStateVariableEntry -ArgumentList $Variable.Name, $Variable.Value, $null) )
|
|
}
|
|
}
|
|
}
|
|
if ($ImportModules)
|
|
{
|
|
if($UserModules.count -gt 0)
|
|
{
|
|
foreach($ModulePath in $UserModules)
|
|
{
|
|
$sessionstate.ImportPSModule($ModulePath)
|
|
}
|
|
}
|
|
if($UserSnapins.count -gt 0)
|
|
{
|
|
foreach($PSSnapin in $UserSnapins)
|
|
{
|
|
[void]$sessionstate.ImportPSSnapIn($PSSnapin, [ref]$null)
|
|
}
|
|
}
|
|
}
|
|
|
|
# --------------------------------------------------
|
|
#region - Import Session Functions
|
|
# --------------------------------------------------
|
|
# Import functions from the current session into the RunspacePool sessionstate
|
|
|
|
if($ImportSessionFunctions)
|
|
{
|
|
# Import all session functions into the runspace session state from the current one
|
|
Get-ChildItem -Path Function:\ |
|
|
Where-Object -FilterScript {
|
|
$_.name -notlike '*:*'
|
|
} |
|
|
Select-Object -Property name -ExpandProperty name |
|
|
ForEach-Object -Process {
|
|
# Get the function code
|
|
$Definition = Get-Content -Path "function:\$_" -ErrorAction Stop
|
|
|
|
# Create a sessionstate function with the same name and code
|
|
$SessionStateFunction = New-Object -TypeName System.Management.Automation.Runspaces.SessionStateFunctionEntry -ArgumentList "$_", $Definition
|
|
|
|
# Add the function to the session state
|
|
$sessionstate.Commands.Add($SessionStateFunction)
|
|
}
|
|
}
|
|
#endregion
|
|
|
|
#Create runspace pool
|
|
$runspacepool = [runspacefactory]::CreateRunspacePool(1, $Throttle, $sessionstate, $Host)
|
|
$runspacepool.Open()
|
|
|
|
#Write-Verbose "Creating empty collection to hold runspace jobs"
|
|
$Script:runspaces = New-Object -TypeName System.Collections.ArrayList
|
|
|
|
#If inputObject is bound get a total count and set bound to true
|
|
$bound = $PSBoundParameters.keys -contains 'InputObject'
|
|
if(-not $bound)
|
|
{
|
|
[System.Collections.ArrayList]$allObjects = @()
|
|
}
|
|
|
|
<#
|
|
#write initial log entry
|
|
$log = "" | Select Date, Action, Runtime, Status, Details
|
|
$log.Date = Get-Date
|
|
$log.Action = "Batch processing started"
|
|
$log.Runtime = $null
|
|
$log.Status = "Started"
|
|
$log.Details = $null
|
|
#>
|
|
$timedOutTasks = $false
|
|
|
|
#endregion INIT
|
|
}
|
|
|
|
Process {
|
|
|
|
#add piped objects to all objects or set all objects to bound input object parameter
|
|
if($bound)
|
|
{
|
|
$allObjects = $InputObject
|
|
}
|
|
Else
|
|
{
|
|
[void]$allObjects.add( $InputObject )
|
|
}
|
|
}
|
|
|
|
End {
|
|
|
|
#Use Try/Finally to catch Ctrl+C and clean up.
|
|
Try
|
|
{
|
|
#counts for progress
|
|
$totalCount = $allObjects.count
|
|
$script:completedCount = 0
|
|
$startedCount = 0
|
|
|
|
foreach($object in $allObjects)
|
|
{
|
|
#region add scripts to runspace pool
|
|
|
|
#Create the powershell instance, set verbose if needed, supply the scriptblock and parameters
|
|
$powershell = [powershell]::Create()
|
|
|
|
if ($VerbosePreference -eq 'Continue')
|
|
{
|
|
[void]$powershell.AddScript({
|
|
$VerbosePreference = 'Continue'
|
|
})
|
|
}
|
|
|
|
[void]$powershell.AddScript($ScriptBlock).AddArgument($object)
|
|
|
|
if ($Parameter)
|
|
{
|
|
[void]$powershell.AddArgument($Parameter)
|
|
}
|
|
|
|
# $Using support from Boe Prox
|
|
if ($UsingVariableData)
|
|
{
|
|
Foreach($UsingVariable in $UsingVariableData)
|
|
{
|
|
#Write-Verbose "Adding $($UsingVariable.Name) with value: $($UsingVariable.Value)"
|
|
[void]$powershell.AddArgument($UsingVariable.Value)
|
|
}
|
|
}
|
|
|
|
#Add the runspace into the powershell instance
|
|
$powershell.RunspacePool = $runspacepool
|
|
|
|
#Create a temporary collection for each runspace
|
|
$temp = '' | Select-Object -Property PowerShell, StartTime, object, Runspace
|
|
$temp.PowerShell = $powershell
|
|
$temp.StartTime = Get-Date
|
|
$temp.object = $object
|
|
|
|
#Save the handle output when calling BeginInvoke() that will be used later to end the runspace
|
|
$temp.Runspace = $powershell.BeginInvoke()
|
|
$startedCount++
|
|
|
|
#Add the temp tracking info to $runspaces collection
|
|
#Write-Verbose ( "Adding {0} to collection at {1}" -f $temp.object, $temp.starttime.tostring() )
|
|
$null = $runspaces.Add($temp)
|
|
|
|
#loop through existing runspaces one time
|
|
if($ShowRunpaceErrors){
|
|
Get-RunspaceData
|
|
}else{
|
|
Get-RunspaceData -ErrorAction SilentlyContinue
|
|
}
|
|
|
|
#If we have more running than max queue (used to control timeout accuracy)
|
|
#Script scope resolves odd PowerShell 2 issue
|
|
$firstRun = $true
|
|
while ($runspaces.count -ge $script:MaxQueue)
|
|
{
|
|
#give verbose output
|
|
if($firstRun)
|
|
{
|
|
#Write-Verbose "$($runspaces.count) items running - exceeded $Script:MaxQueue limit."
|
|
}
|
|
$firstRun = $false
|
|
|
|
#run get-runspace data and sleep for a short while
|
|
Get-RunspaceData
|
|
Start-Sleep -Milliseconds $SleepTimer
|
|
}
|
|
|
|
#endregion add scripts to runspace pool
|
|
}
|
|
|
|
#Write-Verbose ( "Finish processing the remaining runspace jobs: {0}" -f ( @($runspaces | Where {$_.Runspace -ne $Null}).Count) )
|
|
Get-RunspaceData -wait
|
|
|
|
if (-not $Quiet)
|
|
{
|
|
Write-Progress -Activity 'Running Query' -Status 'Starting threads' -Completed
|
|
}
|
|
}
|
|
Finally
|
|
{
|
|
#Close the runspace pool, unless we specified no close on timeout and something timed out
|
|
if ( ($timedOutTasks -eq $false) -or ( ($timedOutTasks -eq $true) -and ($NoCloseOnTimeout -eq $false) ) )
|
|
{
|
|
If (-not($SuppressVerbose)){
|
|
Write-Verbose -Message 'Closing the runspace pool'
|
|
}
|
|
$runspacepool.close()
|
|
}
|
|
|
|
#collect garbage
|
|
[gc]::Collect()
|
|
}
|
|
}
|
|
}
|
|
|
|
# Source: https://stackoverflow.com/questions/35116636/bit-shifting-in-powershell-2-0
|
|
function Convert-BitShift {
|
|
param (
|
|
[Parameter(Position = 0, Mandatory = $True)]
|
|
[int] $Number,
|
|
|
|
[Parameter(ParameterSetName = 'Left', Mandatory = $False)]
|
|
[int] $Left,
|
|
|
|
[Parameter(ParameterSetName = 'Right', Mandatory = $False)]
|
|
[int] $Right
|
|
)
|
|
|
|
$shift = 0
|
|
if ($PSCmdlet.ParameterSetName -eq 'Left')
|
|
{
|
|
$shift = $Left
|
|
}
|
|
else
|
|
{
|
|
$shift = -$Right
|
|
}
|
|
|
|
return [math]::Floor($Number * [math]::Pow(2,$shift))
|
|
}
|
|
|
|
function New-InMemoryModule
|
|
{
|
|
|
|
Param
|
|
(
|
|
[Parameter(Position = 0)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$ModuleName = [Guid]::NewGuid().ToString()
|
|
)
|
|
|
|
$LoadedAssemblies = [AppDomain]::CurrentDomain.GetAssemblies()
|
|
|
|
ForEach ($Assembly in $LoadedAssemblies) {
|
|
if ($Assembly.FullName -and ($Assembly.FullName.Split(',')[0] -eq $ModuleName)) {
|
|
return $Assembly
|
|
}
|
|
}
|
|
|
|
$DynAssembly = New-Object Reflection.AssemblyName($ModuleName)
|
|
$Domain = [AppDomain]::CurrentDomain
|
|
$AssemblyBuilder = $Domain.DefineDynamicAssembly($DynAssembly, 'Run')
|
|
$ModuleBuilder = $AssemblyBuilder.DefineDynamicModule($ModuleName, $False)
|
|
|
|
return $ModuleBuilder
|
|
}
|
|
|
|
function func
|
|
{
|
|
Param
|
|
(
|
|
[Parameter(Position = 0, Mandatory = $True)]
|
|
[String]
|
|
$DllName,
|
|
|
|
[Parameter(Position = 1, Mandatory = $True)]
|
|
[String]
|
|
$FunctionName,
|
|
|
|
[Parameter(Position = 2, Mandatory = $True)]
|
|
[Type]
|
|
$ReturnType,
|
|
|
|
[Parameter(Position = 3)]
|
|
[Type[]]
|
|
$ParameterTypes,
|
|
|
|
[Parameter(Position = 4)]
|
|
[Runtime.InteropServices.CallingConvention]
|
|
$NativeCallingConvention,
|
|
|
|
[Parameter(Position = 5)]
|
|
[Runtime.InteropServices.CharSet]
|
|
$Charset,
|
|
|
|
[Switch]
|
|
$SetLastError
|
|
)
|
|
|
|
$Properties = @{
|
|
DllName = $DllName
|
|
FunctionName = $FunctionName
|
|
ReturnType = $ReturnType
|
|
}
|
|
|
|
if ($ParameterTypes) { $Properties['ParameterTypes'] = $ParameterTypes }
|
|
if ($NativeCallingConvention) { $Properties['NativeCallingConvention'] = $NativeCallingConvention }
|
|
if ($Charset) { $Properties['Charset'] = $Charset }
|
|
if ($SetLastError) { $Properties['SetLastError'] = $SetLastError }
|
|
|
|
New-Object PSObject -Property $Properties
|
|
}
|
|
|
|
function Add-Win32Type
|
|
{
|
|
[OutputType([Hashtable])]
|
|
Param(
|
|
[Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
|
|
[String]
|
|
$DllName,
|
|
|
|
[Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
|
|
[String]
|
|
$FunctionName,
|
|
|
|
[Parameter(Mandatory = $True, ValueFromPipelineByPropertyName = $True)]
|
|
[Type]
|
|
$ReturnType,
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName = $True)]
|
|
[Type[]]
|
|
$ParameterTypes,
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName = $True)]
|
|
[Runtime.InteropServices.CallingConvention]
|
|
$NativeCallingConvention = [Runtime.InteropServices.CallingConvention]::StdCall,
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName = $True)]
|
|
[Runtime.InteropServices.CharSet]
|
|
$Charset = [Runtime.InteropServices.CharSet]::Auto,
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName = $True)]
|
|
[Switch]
|
|
$SetLastError,
|
|
|
|
[Parameter(Mandatory = $True)]
|
|
[ValidateScript({($_ -is [Reflection.Emit.ModuleBuilder]) -or ($_ -is [Reflection.Assembly])})]
|
|
$Module,
|
|
|
|
[ValidateNotNull()]
|
|
[String]
|
|
$Namespace = ''
|
|
)
|
|
|
|
BEGIN
|
|
{
|
|
$TypeHash = @{}
|
|
}
|
|
|
|
PROCESS
|
|
{
|
|
if ($Module -is [Reflection.Assembly])
|
|
{
|
|
if ($Namespace)
|
|
{
|
|
$TypeHash[$DllName] = $Module.GetType("$Namespace.$DllName")
|
|
}
|
|
else
|
|
{
|
|
$TypeHash[$DllName] = $Module.GetType($DllName)
|
|
}
|
|
}
|
|
else
|
|
{
|
|
# Define one type for each DLL
|
|
if (!$TypeHash.ContainsKey($DllName))
|
|
{
|
|
if ($Namespace)
|
|
{
|
|
$TypeHash[$DllName] = $Module.DefineType("$Namespace.$DllName", 'Public,BeforeFieldInit')
|
|
}
|
|
else
|
|
{
|
|
$TypeHash[$DllName] = $Module.DefineType($DllName, 'Public,BeforeFieldInit')
|
|
}
|
|
}
|
|
|
|
$Method = $TypeHash[$DllName].DefineMethod(
|
|
$FunctionName,
|
|
'Public,Static,PinvokeImpl',
|
|
$ReturnType,
|
|
$ParameterTypes)
|
|
|
|
# Make each ByRef parameter an Out parameter
|
|
$i = 1
|
|
ForEach($Parameter in $ParameterTypes)
|
|
{
|
|
if ($Parameter.IsByRef)
|
|
{
|
|
[void] $Method.DefineParameter($i, 'Out', $Null)
|
|
}
|
|
|
|
$i++
|
|
}
|
|
|
|
$DllImport = [Runtime.InteropServices.DllImportAttribute]
|
|
$SetLastErrorField = $DllImport.GetField('SetLastError')
|
|
$CallingConventionField = $DllImport.GetField('CallingConvention')
|
|
$CharsetField = $DllImport.GetField('CharSet')
|
|
if ($SetLastError) { $SLEValue = $True } else { $SLEValue = $False }
|
|
|
|
# Equivalent to C# version of [DllImport(DllName)]
|
|
$Constructor = [Runtime.InteropServices.DllImportAttribute].GetConstructor([String])
|
|
$DllImportAttribute = New-Object Reflection.Emit.CustomAttributeBuilder($Constructor,
|
|
$DllName, [Reflection.PropertyInfo[]] @(), [Object[]] @(),
|
|
[Reflection.FieldInfo[]] @($SetLastErrorField, $CallingConventionField, $CharsetField),
|
|
[Object[]] @($SLEValue, ([Runtime.InteropServices.CallingConvention] $NativeCallingConvention), ([Runtime.InteropServices.CharSet] $Charset)))
|
|
|
|
$Method.SetCustomAttribute($DllImportAttribute)
|
|
}
|
|
}
|
|
|
|
END
|
|
{
|
|
if ($Module -is [Reflection.Assembly])
|
|
{
|
|
return $TypeHash
|
|
}
|
|
|
|
$ReturnTypes = @{}
|
|
|
|
ForEach ($Key in $TypeHash.Keys)
|
|
{
|
|
$Type = $TypeHash[$Key].CreateType()
|
|
|
|
$ReturnTypes[$Key] = $Type
|
|
}
|
|
|
|
return $ReturnTypes
|
|
}
|
|
}
|
|
|
|
|
|
function psenum
|
|
{
|
|
[OutputType([Type])]
|
|
Param
|
|
(
|
|
[Parameter(Position = 0, Mandatory = $True)]
|
|
[ValidateScript({($_ -is [Reflection.Emit.ModuleBuilder]) -or ($_ -is [Reflection.Assembly])})]
|
|
$Module,
|
|
|
|
[Parameter(Position = 1, Mandatory = $True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$FullName,
|
|
|
|
[Parameter(Position = 2, Mandatory = $True)]
|
|
[Type]
|
|
$Type,
|
|
|
|
[Parameter(Position = 3, Mandatory = $True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[Hashtable]
|
|
$EnumElements,
|
|
|
|
[Switch]
|
|
$Bitfield
|
|
)
|
|
|
|
if ($Module -is [Reflection.Assembly])
|
|
{
|
|
return ($Module.GetType($FullName))
|
|
}
|
|
|
|
$EnumType = $Type -as [Type]
|
|
|
|
$EnumBuilder = $Module.DefineEnum($FullName, 'Public', $EnumType)
|
|
|
|
if ($Bitfield)
|
|
{
|
|
$FlagsConstructor = [FlagsAttribute].GetConstructor(@())
|
|
$FlagsCustomAttribute = New-Object Reflection.Emit.CustomAttributeBuilder($FlagsConstructor, @())
|
|
$EnumBuilder.SetCustomAttribute($FlagsCustomAttribute)
|
|
}
|
|
|
|
ForEach ($Key in $EnumElements.Keys)
|
|
{
|
|
# Apply the specified enum type to each element
|
|
$Null = $EnumBuilder.DefineLiteral($Key, $EnumElements[$Key] -as $EnumType)
|
|
}
|
|
|
|
$EnumBuilder.CreateType()
|
|
}
|
|
|
|
function field
|
|
{
|
|
Param
|
|
(
|
|
[Parameter(Position = 0, Mandatory = $True)]
|
|
[UInt16]
|
|
$Position,
|
|
|
|
[Parameter(Position = 1, Mandatory = $True)]
|
|
[Type]
|
|
$Type,
|
|
|
|
[Parameter(Position = 2)]
|
|
[UInt16]
|
|
$Offset,
|
|
|
|
[Object[]]
|
|
$MarshalAs
|
|
)
|
|
|
|
@{
|
|
Position = $Position
|
|
Type = $Type -as [Type]
|
|
Offset = $Offset
|
|
MarshalAs = $MarshalAs
|
|
}
|
|
}
|
|
|
|
function struct
|
|
{
|
|
[OutputType([Type])]
|
|
Param
|
|
(
|
|
[Parameter(Position = 1, Mandatory = $True)]
|
|
[ValidateScript({($_ -is [Reflection.Emit.ModuleBuilder]) -or ($_ -is [Reflection.Assembly])})]
|
|
$Module,
|
|
|
|
[Parameter(Position = 2, Mandatory = $True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$FullName,
|
|
|
|
[Parameter(Position = 3, Mandatory = $True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[Hashtable]
|
|
$StructFields,
|
|
|
|
[Reflection.Emit.PackingSize]
|
|
$PackingSize = [Reflection.Emit.PackingSize]::Unspecified,
|
|
|
|
[Switch]
|
|
$ExplicitLayout
|
|
)
|
|
|
|
if ($Module -is [Reflection.Assembly])
|
|
{
|
|
return ($Module.GetType($FullName))
|
|
}
|
|
|
|
[Reflection.TypeAttributes] $StructAttributes = 'AnsiClass,
|
|
Class,
|
|
Public,
|
|
Sealed,
|
|
BeforeFieldInit'
|
|
|
|
if ($ExplicitLayout)
|
|
{
|
|
$StructAttributes = $StructAttributes -bor [Reflection.TypeAttributes]::ExplicitLayout
|
|
}
|
|
else
|
|
{
|
|
$StructAttributes = $StructAttributes -bor [Reflection.TypeAttributes]::SequentialLayout
|
|
}
|
|
|
|
$StructBuilder = $Module.DefineType($FullName, $StructAttributes, [ValueType], $PackingSize)
|
|
$ConstructorInfo = [Runtime.InteropServices.MarshalAsAttribute].GetConstructors()[0]
|
|
$SizeConst = @([Runtime.InteropServices.MarshalAsAttribute].GetField('SizeConst'))
|
|
|
|
$Fields = New-Object Hashtable[]($StructFields.Count)
|
|
|
|
# Sort each field according to the orders specified
|
|
# Unfortunately, PSv2 doesn't have the luxury of the
|
|
# hashtable [Ordered] accelerator.
|
|
ForEach ($Field in $StructFields.Keys)
|
|
{
|
|
$Index = $StructFields[$Field]['Position']
|
|
$Fields[$Index] = @{FieldName = $Field; Properties = $StructFields[$Field]}
|
|
}
|
|
|
|
ForEach ($Field in $Fields)
|
|
{
|
|
$FieldName = $Field['FieldName']
|
|
$FieldProp = $Field['Properties']
|
|
|
|
$Offset = $FieldProp['Offset']
|
|
$Type = $FieldProp['Type']
|
|
$MarshalAs = $FieldProp['MarshalAs']
|
|
|
|
$NewField = $StructBuilder.DefineField($FieldName, $Type, 'Public')
|
|
|
|
if ($MarshalAs)
|
|
{
|
|
$UnmanagedType = $MarshalAs[0] -as ([Runtime.InteropServices.UnmanagedType])
|
|
if ($MarshalAs[1])
|
|
{
|
|
$Size = $MarshalAs[1]
|
|
$AttribBuilder = New-Object Reflection.Emit.CustomAttributeBuilder($ConstructorInfo,
|
|
$UnmanagedType, $SizeConst, @($Size))
|
|
}
|
|
else
|
|
{
|
|
$AttribBuilder = New-Object Reflection.Emit.CustomAttributeBuilder($ConstructorInfo, [Object[]] @($UnmanagedType))
|
|
}
|
|
|
|
$NewField.SetCustomAttribute($AttribBuilder)
|
|
}
|
|
|
|
if ($ExplicitLayout) { $NewField.SetOffset($Offset) }
|
|
}
|
|
|
|
# Make the struct aware of its own size.
|
|
# No more having to call [Runtime.InteropServices.Marshal]::SizeOf!
|
|
$SizeMethod = $StructBuilder.DefineMethod('GetSize',
|
|
'Public, Static',
|
|
[Int],
|
|
[Type[]] @())
|
|
$ILGenerator = $SizeMethod.GetILGenerator()
|
|
# Thanks for the help, Jason Shirk!
|
|
$ILGenerator.Emit([Reflection.Emit.OpCodes]::Ldtoken, $StructBuilder)
|
|
$ILGenerator.Emit([Reflection.Emit.OpCodes]::Call,
|
|
[Type].GetMethod('GetTypeFromHandle'))
|
|
$ILGenerator.Emit([Reflection.Emit.OpCodes]::Call,
|
|
[Runtime.InteropServices.Marshal].GetMethod('SizeOf', [Type[]] @([Type])))
|
|
$ILGenerator.Emit([Reflection.Emit.OpCodes]::Ret)
|
|
|
|
# Allow for explicit casting from an IntPtr
|
|
# No more having to call [Runtime.InteropServices.Marshal]::PtrToStructure!
|
|
$ImplicitConverter = $StructBuilder.DefineMethod('op_Implicit',
|
|
'PrivateScope, Public, Static, HideBySig, SpecialName',
|
|
$StructBuilder,
|
|
[Type[]] @([IntPtr]))
|
|
$ILGenerator2 = $ImplicitConverter.GetILGenerator()
|
|
$ILGenerator2.Emit([Reflection.Emit.OpCodes]::Nop)
|
|
$ILGenerator2.Emit([Reflection.Emit.OpCodes]::Ldarg_0)
|
|
$ILGenerator2.Emit([Reflection.Emit.OpCodes]::Ldtoken, $StructBuilder)
|
|
$ILGenerator2.Emit([Reflection.Emit.OpCodes]::Call,
|
|
[Type].GetMethod('GetTypeFromHandle'))
|
|
$ILGenerator2.Emit([Reflection.Emit.OpCodes]::Call,
|
|
[Runtime.InteropServices.Marshal].GetMethod('PtrToStructure', [Type[]] @([IntPtr], [Type])))
|
|
$ILGenerator2.Emit([Reflection.Emit.OpCodes]::Unbox_Any, $StructBuilder)
|
|
$ILGenerator2.Emit([Reflection.Emit.OpCodes]::Ret)
|
|
|
|
$StructBuilder.CreateType()
|
|
}
|
|
|
|
filter Get-IniContent {
|
|
[CmdletBinding()]
|
|
Param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True, ValueFromPipelineByPropertyName=$True)]
|
|
[Alias('FullName')]
|
|
[ValidateScript({ Test-Path -Path $_ })]
|
|
[String[]]
|
|
$Path
|
|
)
|
|
|
|
ForEach($TargetPath in $Path) {
|
|
$IniObject = @{}
|
|
Switch -Regex -File $TargetPath {
|
|
"^\[(.+)\]" # Section
|
|
{
|
|
$Section = $matches[1].Trim()
|
|
$IniObject[$Section] = @{}
|
|
$CommentCount = 0
|
|
}
|
|
"^(;.*)$" # Comment
|
|
{
|
|
$Value = $matches[1].Trim()
|
|
$CommentCount = $CommentCount + 1
|
|
$Name = 'Comment' + $CommentCount
|
|
$IniObject[$Section][$Name] = $Value
|
|
}
|
|
"(.+?)\s*=(.*)" # Key
|
|
{
|
|
$Name, $Value = $matches[1..2]
|
|
$Name = $Name.Trim()
|
|
$Values = $Value.split(',') | ForEach-Object {$_.Trim()}
|
|
if($Values -isnot [System.Array]) {$Values = @($Values)}
|
|
$IniObject[$Section][$Name] = $Values
|
|
}
|
|
}
|
|
$IniObject
|
|
}
|
|
}
|
|
|
|
filter Export-PowerViewCSV {
|
|
|
|
Param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True, ValueFromPipelineByPropertyName=$True)]
|
|
[System.Management.Automation.PSObject[]]
|
|
$InputObject,
|
|
|
|
[Parameter(Mandatory=$True, Position=0)]
|
|
[String]
|
|
[ValidateNotNullOrEmpty()]
|
|
$OutFile
|
|
)
|
|
|
|
$ObjectCSV = $InputObject | ConvertTo-Csv -NoTypeInformation
|
|
|
|
# mutex so threaded code doesn't stomp on the output file
|
|
$Mutex = New-Object System.Threading.Mutex $False,'CSVMutex';
|
|
$Null = $Mutex.WaitOne()
|
|
|
|
if (Test-Path -Path $OutFile) {
|
|
# hack to skip the first line of output if the file already exists
|
|
$ObjectCSV | ForEach-Object { $Start=$True }{ if ($Start) {$Start=$False} else {$_} } | Out-File -Encoding 'ASCII' -Append -FilePath $OutFile
|
|
}
|
|
else {
|
|
$ObjectCSV | Out-File -Encoding 'ASCII' -Append -FilePath $OutFile
|
|
}
|
|
|
|
$Mutex.ReleaseMutex()
|
|
}
|
|
|
|
filter Get-IPAddress {
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0, ValueFromPipeline=$True)]
|
|
[Alias('HostName')]
|
|
[String]
|
|
$ComputerName = $Env:ComputerName
|
|
)
|
|
|
|
try {
|
|
# extract the computer name from whatever object was passed on the pipeline
|
|
$Computer = $ComputerName | Get-NameField
|
|
|
|
# get the IP resolution of this specified hostname
|
|
@(([Net.Dns]::GetHostEntry($Computer)).AddressList) | ForEach-Object {
|
|
if ($_.AddressFamily -eq 'InterNetwork') {
|
|
$Out = New-Object PSObject
|
|
$Out | Add-Member Noteproperty 'ComputerName' $Computer
|
|
$Out | Add-Member Noteproperty 'IPAddress' $_.IPAddressToString
|
|
$Out
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose -Message 'Could not resolve host to an IP Address.'
|
|
}
|
|
}
|
|
|
|
filter Convert-NameToSid {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
[String]
|
|
[Alias('Name')]
|
|
$ObjectName,
|
|
|
|
[String]
|
|
$Domain
|
|
)
|
|
|
|
$ObjectName = $ObjectName -Replace "/","\"
|
|
|
|
if($ObjectName.Contains("\")) {
|
|
# if we get a DOMAIN\user format, auto convert it
|
|
$Domain = $ObjectName.Split("\")[0]
|
|
$ObjectName = $ObjectName.Split("\")[1]
|
|
}
|
|
elseif(-not $Domain) {
|
|
$Domain = (Get-ThisThingDomain).Name
|
|
}
|
|
|
|
try {
|
|
$Obj = (New-Object System.Security.Principal.NTAccount($Domain, $ObjectName))
|
|
$SID = $Obj.Translate([System.Security.Principal.SecurityIdentifier]).Value
|
|
|
|
$Out = New-Object PSObject
|
|
$Out | Add-Member Noteproperty 'ObjectName' $ObjectName
|
|
$Out | Add-Member Noteproperty 'SID' $SID
|
|
$Out
|
|
}
|
|
catch {
|
|
Write-Verbose "Invalid object/name: $Domain\$ObjectName"
|
|
$Null
|
|
}
|
|
}
|
|
|
|
filter Convert-SidToName {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
[String]
|
|
[ValidatePattern('^S-1-.*')]
|
|
$SID
|
|
)
|
|
|
|
try {
|
|
$SID2 = $SID.trim('*')
|
|
|
|
# try to resolve any built-in SIDs first
|
|
# from https://support.microsoft.com/en-us/kb/243330
|
|
Switch ($SID2) {
|
|
'S-1-0' { 'Null Authority' }
|
|
'S-1-0-0' { 'Nobody' }
|
|
'S-1-1' { 'World Authority' }
|
|
'S-1-1-0' { 'Everyone' }
|
|
'S-1-2' { 'Local Authority' }
|
|
'S-1-2-0' { 'Local' }
|
|
'S-1-2-1' { 'Console Logon ' }
|
|
'S-1-3' { 'Creator Authority' }
|
|
'S-1-3-0' { 'Creator Owner' }
|
|
'S-1-3-1' { 'Creator Group' }
|
|
'S-1-3-2' { 'Creator Owner Server' }
|
|
'S-1-3-3' { 'Creator Group Server' }
|
|
'S-1-3-4' { 'Owner Rights' }
|
|
'S-1-4' { 'Non-unique Authority' }
|
|
'S-1-5' { 'NT Authority' }
|
|
'S-1-5-1' { 'Dialup' }
|
|
'S-1-5-2' { 'Network' }
|
|
'S-1-5-3' { 'Batch' }
|
|
'S-1-5-4' { 'Interactive' }
|
|
'S-1-5-6' { 'Service' }
|
|
'S-1-5-7' { 'Anonymous' }
|
|
'S-1-5-8' { 'Proxy' }
|
|
'S-1-5-9' { 'Enterprise Domain Controllers' }
|
|
'S-1-5-10' { 'Principal Self' }
|
|
'S-1-5-11' { 'Authenticated Users' }
|
|
'S-1-5-12' { 'Restricted Code' }
|
|
'S-1-5-13' { 'Terminal Server Users' }
|
|
'S-1-5-14' { 'Remote Interactive Logon' }
|
|
'S-1-5-15' { 'This Organization ' }
|
|
'S-1-5-17' { 'This Organization ' }
|
|
'S-1-5-18' { 'Local System' }
|
|
'S-1-5-19' { 'NT Authority' }
|
|
'S-1-5-20' { 'NT Authority' }
|
|
'S-1-5-80-0' { 'All Services ' }
|
|
'S-1-5-32-544' { 'BUILTIN\Administrators' }
|
|
'S-1-5-32-545' { 'BUILTIN\Users' }
|
|
'S-1-5-32-546' { 'BUILTIN\Guests' }
|
|
'S-1-5-32-547' { 'BUILTIN\Power Users' }
|
|
'S-1-5-32-548' { 'BUILTIN\Account Operators' }
|
|
'S-1-5-32-549' { 'BUILTIN\Server Operators' }
|
|
'S-1-5-32-550' { 'BUILTIN\Print Operators' }
|
|
'S-1-5-32-551' { 'BUILTIN\Backup Operators' }
|
|
'S-1-5-32-552' { 'BUILTIN\Replicators' }
|
|
'S-1-5-32-554' { 'BUILTIN\Pre-Windows 2000 Compatible Access' }
|
|
'S-1-5-32-555' { 'BUILTIN\Remote Desktop Users' }
|
|
'S-1-5-32-556' { 'BUILTIN\Network Configuration Operators' }
|
|
'S-1-5-32-557' { 'BUILTIN\Incoming Forest Trust Builders' }
|
|
'S-1-5-32-558' { 'BUILTIN\Performance Monitor Users' }
|
|
'S-1-5-32-559' { 'BUILTIN\Performance Log Users' }
|
|
'S-1-5-32-560' { 'BUILTIN\Windows Authorization Access Group' }
|
|
'S-1-5-32-561' { 'BUILTIN\Terminal Server License Servers' }
|
|
'S-1-5-32-562' { 'BUILTIN\Distributed COM Users' }
|
|
'S-1-5-32-569' { 'BUILTIN\Cryptographic Operators' }
|
|
'S-1-5-32-573' { 'BUILTIN\Event Log Readers' }
|
|
'S-1-5-32-574' { 'BUILTIN\Certificate Service DCOM Access' }
|
|
'S-1-5-32-575' { 'BUILTIN\RDS Remote Access Servers' }
|
|
'S-1-5-32-576' { 'BUILTIN\RDS Endpoint Servers' }
|
|
'S-1-5-32-577' { 'BUILTIN\RDS Management Servers' }
|
|
'S-1-5-32-578' { 'BUILTIN\Hyper-V Administrators' }
|
|
'S-1-5-32-579' { 'BUILTIN\Access Control Assistance Operators' }
|
|
'S-1-5-32-580' { 'BUILTIN\Access Control Assistance Operators' }
|
|
Default {
|
|
$Obj = (New-Object System.Security.Principal.SecurityIdentifier($SID2))
|
|
$Obj.Translate( [System.Security.Principal.NTAccount]).Value
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose "Invalid SID: $SID"
|
|
$SID
|
|
}
|
|
}
|
|
|
|
filter Convert-ADName {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
[String]
|
|
$ObjectName,
|
|
|
|
[String]
|
|
[ValidateSet("NT4","Simple","Canonical")]
|
|
$InputType,
|
|
|
|
[String]
|
|
[ValidateSet("NT4","Simple","Canonical")]
|
|
$OutputType
|
|
)
|
|
|
|
$NameTypes = @{
|
|
'Canonical' = 2
|
|
'NT4' = 3
|
|
'Simple' = 5
|
|
}
|
|
|
|
if(-not $PSBoundParameters['InputType']) {
|
|
if( ($ObjectName.split('/')).Count -eq 2 ) {
|
|
$ObjectName = $ObjectName.replace('/', '\')
|
|
}
|
|
|
|
if($ObjectName -match "^[A-Za-z]+\\[A-Za-z ]+") {
|
|
$InputType = 'NT4'
|
|
}
|
|
elseif($ObjectName -match "^[A-Za-z ]+@[A-Za-z\.]+") {
|
|
$InputType = 'Simple'
|
|
}
|
|
elseif($ObjectName -match "^[A-Za-z\.]+/[A-Za-z]+/[A-Za-z/ ]+") {
|
|
$InputType = 'Canonical'
|
|
}
|
|
else {
|
|
#Write-Warning "Can not identify InType for $ObjectName"
|
|
return $ObjectName
|
|
}
|
|
}
|
|
elseif($InputType -eq 'NT4') {
|
|
$ObjectName = $ObjectName.replace('/', '\')
|
|
}
|
|
|
|
if(-not $PSBoundParameters['OutputType']) {
|
|
$OutputType = Switch($InputType) {
|
|
'NT4' {'Canonical'}
|
|
'Simple' {'NT4'}
|
|
'Canonical' {'NT4'}
|
|
}
|
|
}
|
|
|
|
# try to extract the domain from the given format
|
|
$Domain = Switch($InputType) {
|
|
'NT4' { $ObjectName.split("\")[0] }
|
|
'Simple' { $ObjectName.split("@")[1] }
|
|
'Canonical' { $ObjectName.split("/")[0] }
|
|
}
|
|
|
|
# Accessor functions to simplify calls to NameTranslate
|
|
function Invoke-Method([__ComObject] $Object, [String] $Method, $Parameters) {
|
|
$Output = $Object.GetType().InvokeMember($Method, "InvokeMethod", $Null, $Object, $Parameters)
|
|
if ( $Output ) { $Output }
|
|
}
|
|
function Set-Property([__ComObject] $Object, [String] $Property, $Parameters) {
|
|
[Void] $Object.GetType().InvokeMember($Property, "SetProperty", $Null, $Object, $Parameters)
|
|
}
|
|
|
|
$Translate = New-Object -ComObject NameTranslate
|
|
|
|
try {
|
|
Invoke-Method $Translate "Init" (1, $Domain)
|
|
}
|
|
catch [System.Management.Automation.MethodInvocationException] {
|
|
Write-Verbose "Error with translate init in Convert-ADName: $_"
|
|
}
|
|
|
|
Set-Property $Translate "ChaseReferral" (0x60)
|
|
|
|
try {
|
|
Invoke-Method $Translate "Set" ($NameTypes[$InputType], $ObjectName)
|
|
(Invoke-Method $Translate "Get" ($NameTypes[$OutputType]))
|
|
}
|
|
catch [System.Management.Automation.MethodInvocationException] {
|
|
Write-Verbose "Error with translate Set/Get in Convert-ADName: $_"
|
|
}
|
|
}
|
|
|
|
function ConvertFrom-UACValue {
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
$Value,
|
|
|
|
[Switch]
|
|
$ShowAll
|
|
)
|
|
|
|
begin {
|
|
# values from https://support.microsoft.com/en-us/kb/305144
|
|
$UACValues = New-Object System.Collections.Specialized.OrderedDictionary
|
|
$UACValues.Add("SCRIPT", 1)
|
|
$UACValues.Add("ACCOUNTDISABLE", 2)
|
|
$UACValues.Add("HOMEDIR_REQUIRED", 8)
|
|
$UACValues.Add("LOCKOUT", 16)
|
|
$UACValues.Add("PASSWD_NOTREQD", 32)
|
|
$UACValues.Add("PASSWD_CANT_CHANGE", 64)
|
|
$UACValues.Add("ENCRYPTED_TEXT_PWD_ALLOWED", 128)
|
|
$UACValues.Add("TEMP_DUPLICATE_ACCOUNT", 256)
|
|
$UACValues.Add("NORMAL_ACCOUNT", 512)
|
|
$UACValues.Add("INTERDOMAIN_TRUST_ACCOUNT", 2048)
|
|
$UACValues.Add("WORKSTATION_TRUST_ACCOUNT", 4096)
|
|
$UACValues.Add("SERVER_TRUST_ACCOUNT", 8192)
|
|
$UACValues.Add("DONT_EXPIRE_PASSWORD", 65536)
|
|
$UACValues.Add("MNS_LOGON_ACCOUNT", 131072)
|
|
$UACValues.Add("SMARTCARD_REQUIRED", 262144)
|
|
$UACValues.Add("TRUSTED_FOR_DELEGATION", 524288)
|
|
$UACValues.Add("NOT_DELEGATED", 1048576)
|
|
$UACValues.Add("USE_DES_KEY_ONLY", 2097152)
|
|
$UACValues.Add("DONT_REQ_PREAUTH", 4194304)
|
|
$UACValues.Add("PASSWORD_EXPIRED", 8388608)
|
|
$UACValues.Add("TRUSTED_TO_AUTH_FOR_DELEGATION", 16777216)
|
|
$UACValues.Add("PARTIAL_SECRETS_ACCOUNT", 67108864)
|
|
}
|
|
|
|
process {
|
|
|
|
$ResultUACValues = New-Object System.Collections.Specialized.OrderedDictionary
|
|
|
|
if($Value -is [Int]) {
|
|
$IntValue = $Value
|
|
}
|
|
elseif ($Value -is [PSCustomObject]) {
|
|
if($Value.useraccountcontrol) {
|
|
$IntValue = $Value.useraccountcontrol
|
|
}
|
|
}
|
|
else {
|
|
#Write-Warning "Invalid object input for -Value : $Value"
|
|
return $Null
|
|
}
|
|
|
|
if($ShowAll) {
|
|
foreach ($UACValue in $UACValues.GetEnumerator()) {
|
|
if( ($IntValue -band $UACValue.Value) -eq $UACValue.Value) {
|
|
$ResultUACValues.Add($UACValue.Name, "$($UACValue.Value)+")
|
|
}
|
|
else {
|
|
$ResultUACValues.Add($UACValue.Name, "$($UACValue.Value)")
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
foreach ($UACValue in $UACValues.GetEnumerator()) {
|
|
if( ($IntValue -band $UACValue.Value) -eq $UACValue.Value) {
|
|
$ResultUACValues.Add($UACValue.Name, "$($UACValue.Value)")
|
|
}
|
|
}
|
|
}
|
|
$ResultUACValues
|
|
}
|
|
}
|
|
|
|
filter Get-Proxy {
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$ComputerName = $ENV:COMPUTERNAME
|
|
)
|
|
|
|
try {
|
|
$Reg = [Microsoft.Win32.RegistryKey]::OpenRemoteBaseKey('CurrentUser', $ComputerName)
|
|
$RegKey = $Reg.OpenSubkey("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Internet Settings")
|
|
$ProxyServer = $RegKey.GetValue('ProxyServer')
|
|
$AutoConfigURL = $RegKey.GetValue('AutoConfigURL')
|
|
|
|
$Wpad = ""
|
|
if($AutoConfigURL -and ($AutoConfigURL -ne "")) {
|
|
try {
|
|
$Wpad = (New-Object Net.Webclient).DownloadString($AutoConfigURL)
|
|
}
|
|
catch {
|
|
#Write-Warning "Error connecting to AutoConfigURL : $AutoConfigURL"
|
|
}
|
|
}
|
|
|
|
if($ProxyServer -or $AutoConfigUrl) {
|
|
|
|
$Properties = @{
|
|
'ProxyServer' = $ProxyServer
|
|
'AutoConfigURL' = $AutoConfigURL
|
|
'Wpad' = $Wpad
|
|
}
|
|
|
|
New-Object -TypeName PSObject -Property $Properties
|
|
}
|
|
else {
|
|
Write-Warning "No proxy settings found for $ComputerName"
|
|
}
|
|
}
|
|
catch {
|
|
Write-Warning "Error enumerating proxy settings for $ComputerName : $_"
|
|
}
|
|
}
|
|
|
|
function Request-SPNTicket {
|
|
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(Mandatory=$True, ValueFromPipelineByPropertyName = $True)]
|
|
[Alias('ServicePrincipalName')]
|
|
[String[]]
|
|
$SPN,
|
|
|
|
[Alias('EncryptedPart')]
|
|
[Switch]
|
|
$EncPart
|
|
)
|
|
|
|
begin {
|
|
Add-Type -AssemblyName System.IdentityModel
|
|
}
|
|
|
|
process {
|
|
ForEach($UserSPN in $SPN) {
|
|
Write-Verbose "Requesting ticket for: $UserSPN"
|
|
if (!$EncPart) {
|
|
New-Object System.IdentityModel.Tokens.KerberosRequestorSecurityToken -ArgumentList $UserSPN
|
|
}
|
|
else {
|
|
$Ticket = New-Object System.IdentityModel.Tokens.KerberosRequestorSecurityToken -ArgumentList $UserSPN
|
|
$TicketByteStream = $Ticket.GetRequest()
|
|
if ($TicketByteStream)
|
|
{
|
|
$TicketHexStream = [System.BitConverter]::ToString($TicketByteStream) -replace "-"
|
|
[System.Collections.ArrayList]$Parts = ($TicketHexStream -replace '^(.*?)04820...(.*)','$2') -Split "A48201"
|
|
$Parts.RemoveAt($Parts.Count - 1)
|
|
$Parts -join "A48201"
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-PathAcl {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
[String]
|
|
$Path,
|
|
|
|
[Switch]
|
|
$Recurse
|
|
)
|
|
|
|
begin {
|
|
|
|
function Convert-FileRight {
|
|
|
|
# From http://stackoverflow.com/questions/28029872/retrieving-security-descriptor-and-getting-number-for-filesystemrights
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Int]
|
|
$FSR
|
|
)
|
|
|
|
$AccessMask = @{
|
|
[uint32]'0x80000000' = 'GenericRead'
|
|
[uint32]'0x40000000' = 'GenericWrite'
|
|
[uint32]'0x20000000' = 'GenericExecute'
|
|
[uint32]'0x10000000' = 'GenericAll'
|
|
[uint32]'0x02000000' = 'MaximumAllowed'
|
|
[uint32]'0x01000000' = 'AccessSystemSecurity'
|
|
[uint32]'0x00100000' = 'Synchronize'
|
|
[uint32]'0x00080000' = 'WriteOwner'
|
|
[uint32]'0x00040000' = 'WriteDAC'
|
|
[uint32]'0x00020000' = 'ReadControl'
|
|
[uint32]'0x00010000' = 'Delete'
|
|
[uint32]'0x00000100' = 'WriteAttributes'
|
|
[uint32]'0x00000080' = 'ReadAttributes'
|
|
[uint32]'0x00000040' = 'DeleteChild'
|
|
[uint32]'0x00000020' = 'Execute/Traverse'
|
|
[uint32]'0x00000010' = 'WriteExtendedAttributes'
|
|
[uint32]'0x00000008' = 'ReadExtendedAttributes'
|
|
[uint32]'0x00000004' = 'AppendData/AddSubdirectory'
|
|
[uint32]'0x00000002' = 'WriteData/AddFile'
|
|
[uint32]'0x00000001' = 'ReadData/ListDirectory'
|
|
}
|
|
|
|
$SimplePermissions = @{
|
|
[uint32]'0x1f01ff' = 'FullControl'
|
|
[uint32]'0x0301bf' = 'Modify'
|
|
[uint32]'0x0200a9' = 'ReadAndExecute'
|
|
[uint32]'0x02019f' = 'ReadAndWrite'
|
|
[uint32]'0x020089' = 'Read'
|
|
[uint32]'0x000116' = 'Write'
|
|
}
|
|
|
|
$Permissions = @()
|
|
|
|
# get simple permission
|
|
$Permissions += $SimplePermissions.Keys | % {
|
|
if (($FSR -band $_) -eq $_) {
|
|
$SimplePermissions[$_]
|
|
$FSR = $FSR -band (-not $_)
|
|
}
|
|
}
|
|
|
|
# get remaining extended permissions
|
|
$Permissions += $AccessMask.Keys |
|
|
? { $FSR -band $_ } |
|
|
% { $AccessMask[$_] }
|
|
|
|
($Permissions | ?{$_}) -join ","
|
|
}
|
|
}
|
|
|
|
process {
|
|
|
|
try {
|
|
$ACL = Get-Acl -Path $Path
|
|
|
|
[String]$PathOwner = $ACL.Owner
|
|
|
|
$ACL.GetAccessRules($true,$true,[System.Security.Principal.SecurityIdentifier]) | ForEach-Object {
|
|
|
|
$Names = @()
|
|
if ($_.IdentityReference -match '^S-1-5-21-[0-9]+-[0-9]+-[0-9]+-[0-9]+') {
|
|
$Object = Get-ADObject -SID $_.IdentityReference
|
|
$Names = @()
|
|
$SIDs = @($Object.objectsid)
|
|
|
|
if ($Recurse -and (@('268435456','268435457','536870912','536870913') -contains $Object.samAccountType)) {
|
|
$SIDs += Get-ThisThingGroupMember -SID $Object.objectsid | Select-Object -ExpandProperty MemberSid
|
|
}
|
|
|
|
$SIDs | ForEach-Object {
|
|
$Names += ,@($_, (Convert-SidToName $_))
|
|
}
|
|
}
|
|
else {
|
|
$Names += ,@($_.IdentityReference.Value, (Convert-SidToName $_.IdentityReference.Value))
|
|
}
|
|
|
|
ForEach($Name in $Names) {
|
|
$Out = New-Object PSObject
|
|
$Out | Add-Member Noteproperty 'Path' $Path
|
|
$Out | Add-Member Noteproperty 'PathOwner' $PathOwner
|
|
$Out | Add-Member Noteproperty 'FileSystemRights' (Convert-FileRight -FSR $_.FileSystemRights.value__)
|
|
$Out | Add-Member Noteproperty 'IdentityReference' $Name[1]
|
|
$Out | Add-Member Noteproperty 'IdentitySID' $Name[0]
|
|
$Out | Add-Member Noteproperty 'AccessControlType' $_.AccessControlType
|
|
$Out
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
#Write-Warning $_
|
|
}
|
|
}
|
|
}
|
|
|
|
filter Get-NameField {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(ValueFromPipeline = $True, ValueFromPipelineByPropertyName = $True)]
|
|
[Object]
|
|
$Object,
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName = $True)]
|
|
[String]
|
|
$DnsHostName,
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName = $True)]
|
|
[String]
|
|
$Name
|
|
)
|
|
|
|
if($PSBoundParameters['DnsHostName']) {
|
|
$DnsHostName
|
|
}
|
|
elseif($PSBoundParameters['Name']) {
|
|
$Name
|
|
}
|
|
elseif($Object) {
|
|
if ( [bool]($Object.PSobject.Properties.name -match "dnshostname") ) {
|
|
# objects from Get-ThisThingComputer
|
|
$Object.dnshostname
|
|
}
|
|
elseif ( [bool]($Object.PSobject.Properties.name -match "name") ) {
|
|
# objects from Get-ThisThingDomainController
|
|
$Object.name
|
|
}
|
|
else {
|
|
# strings and catch alls
|
|
$Object
|
|
}
|
|
}
|
|
else {
|
|
return $Null
|
|
}
|
|
}
|
|
|
|
function Convert-LDAPProperty {
|
|
param(
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
$Properties
|
|
)
|
|
|
|
$ObjectProperties = @{}
|
|
|
|
$Properties.PropertyNames | ForEach-Object {
|
|
if (($_ -eq "objectsid") -or ($_ -eq "sidhistory")) {
|
|
# convert the SID to a string
|
|
$ObjectProperties[$_] = (New-Object System.Security.Principal.SecurityIdentifier($Properties[$_][0],0)).Value
|
|
}
|
|
elseif($_ -eq "objectguid") {
|
|
# convert the GUID to a string
|
|
$ObjectProperties[$_] = (New-Object Guid (,$Properties[$_][0])).Guid
|
|
}
|
|
elseif( ($_ -eq "lastlogon") -or ($_ -eq "lastlogontimestamp") -or ($_ -eq "pwdlastset") -or ($_ -eq "lastlogoff") -or ($_ -eq "badPasswordTime") ) {
|
|
# convert timestamps
|
|
if ($Properties[$_][0] -is [System.MarshalByRefObject]) {
|
|
# if we have a System.__ComObject
|
|
$Temp = $Properties[$_][0]
|
|
[Int32]$High = $Temp.GetType().InvokeMember("HighPart", [System.Reflection.BindingFlags]::GetProperty, $null, $Temp, $null)
|
|
[Int32]$Low = $Temp.GetType().InvokeMember("LowPart", [System.Reflection.BindingFlags]::GetProperty, $null, $Temp, $null)
|
|
$ObjectProperties[$_] = ([datetime]::FromFileTime([Int64]("0x{0:x8}{1:x8}" -f $High, $Low)))
|
|
}
|
|
else {
|
|
$ObjectProperties[$_] = ([datetime]::FromFileTime(($Properties[$_][0])))
|
|
}
|
|
}
|
|
elseif($Properties[$_][0] -is [System.MarshalByRefObject]) {
|
|
# try to convert misc com objects
|
|
$Prop = $Properties[$_]
|
|
try {
|
|
$Temp = $Prop[$_][0]
|
|
Write-Verbose $_
|
|
[Int32]$High = $Temp.GetType().InvokeMember("HighPart", [System.Reflection.BindingFlags]::GetProperty, $null, $Temp, $null)
|
|
[Int32]$Low = $Temp.GetType().InvokeMember("LowPart", [System.Reflection.BindingFlags]::GetProperty, $null, $Temp, $null)
|
|
$ObjectProperties[$_] = [Int64]("0x{0:x8}{1:x8}" -f $High, $Low)
|
|
}
|
|
catch {
|
|
$ObjectProperties[$_] = $Prop[$_]
|
|
}
|
|
}
|
|
elseif($Properties[$_].count -eq 1) {
|
|
$ObjectProperties[$_] = $Properties[$_][0]
|
|
}
|
|
else {
|
|
$ObjectProperties[$_] = $Properties[$_]
|
|
}
|
|
}
|
|
|
|
New-Object -TypeName PSObject -Property $ObjectProperties
|
|
}
|
|
|
|
filter Get-DomainSearcher {
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$ADSprefix,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
if(-not $Credential) {
|
|
if(-not $Domain) {
|
|
$Domain = (Get-ThisThingDomain).name
|
|
}
|
|
elseif(-not $DomainController) {
|
|
try {
|
|
# if there's no -DomainController specified, try to pull the primary DC to reflect queries through
|
|
$DomainController = ((Get-ThisThingDomain).PdcRoleOwner).Name
|
|
}
|
|
catch {
|
|
throw "Get-DomainSearcher: Error in retrieving PDC for current domain"
|
|
}
|
|
}
|
|
}
|
|
elseif (-not $DomainController) {
|
|
# if a DC isn't specified
|
|
try {
|
|
$DomainController = ((Get-ThisThingDomain -Credential $Credential).PdcRoleOwner).Name
|
|
}
|
|
catch {
|
|
throw "Get-DomainSearcher: Error in retrieving PDC for current domain"
|
|
}
|
|
|
|
if(!$DomainController) {
|
|
throw "Get-DomainSearcher: Error in retrieving PDC for current domain"
|
|
}
|
|
}
|
|
|
|
$SearchString = "LDAP://"
|
|
|
|
if($DomainController) {
|
|
$SearchString += $DomainController
|
|
if($Domain){
|
|
$SearchString += '/'
|
|
}
|
|
}
|
|
|
|
if($ADSprefix) {
|
|
$SearchString += $ADSprefix + ','
|
|
}
|
|
|
|
if($ADSpath) {
|
|
if($ADSpath -Match '^GC://') {
|
|
# if we're searching the global catalog
|
|
$DN = $AdsPath.ToUpper().Trim('/')
|
|
$SearchString = ''
|
|
}
|
|
else {
|
|
if($ADSpath -match '^LDAP://') {
|
|
if($ADSpath -match "LDAP://.+/.+") {
|
|
$SearchString = ''
|
|
}
|
|
else {
|
|
$ADSpath = $ADSpath.Substring(7)
|
|
}
|
|
}
|
|
$DN = $ADSpath
|
|
}
|
|
}
|
|
else {
|
|
if($Domain -and ($Domain.Trim() -ne "")) {
|
|
$DN = "DC=$($Domain.Replace('.', ',DC='))"
|
|
}
|
|
}
|
|
|
|
$SearchString += $DN
|
|
Write-Verbose "Get-DomainSearcher search string: $SearchString"
|
|
|
|
if($Credential) {
|
|
Write-Verbose "Using alternate credentials for LDAP connection"
|
|
$DomainObject = New-Object DirectoryServices.DirectoryEntry($SearchString, $Credential.UserName, $Credential.GetNetworkCredential().Password)
|
|
$Searcher = New-Object System.DirectoryServices.DirectorySearcher($DomainObject)
|
|
}
|
|
else {
|
|
$Searcher = New-Object System.DirectoryServices.DirectorySearcher([ADSI]$SearchString)
|
|
}
|
|
|
|
$Searcher.PageSize = $PageSize
|
|
$Searcher.CacheResults = $False
|
|
$Searcher
|
|
}
|
|
|
|
filter Convert-DNSRecord {
|
|
param(
|
|
[Parameter(Position=0, ValueFromPipelineByPropertyName=$True, Mandatory=$True)]
|
|
[Byte[]]
|
|
$DNSRecord
|
|
)
|
|
|
|
function Get-Name {
|
|
# modified decodeName from https://raw.githubusercontent.com/mmessano/PowerShell/master/dns-dump.ps1
|
|
[CmdletBinding()]
|
|
param(
|
|
[Byte[]]
|
|
$Raw
|
|
)
|
|
|
|
[Int]$Length = $Raw[0]
|
|
[Int]$Segments = $Raw[1]
|
|
[Int]$Index = 2
|
|
[String]$Name = ""
|
|
|
|
while ($Segments-- -gt 0)
|
|
{
|
|
[Int]$SegmentLength = $Raw[$Index++]
|
|
while ($SegmentLength-- -gt 0) {
|
|
$Name += [Char]$Raw[$Index++]
|
|
}
|
|
$Name += "."
|
|
}
|
|
$Name
|
|
}
|
|
|
|
$RDataLen = [BitConverter]::ToUInt16($DNSRecord, 0)
|
|
$RDataType = [BitConverter]::ToUInt16($DNSRecord, 2)
|
|
$UpdatedAtSerial = [BitConverter]::ToUInt32($DNSRecord, 8)
|
|
|
|
$TTLRaw = $DNSRecord[12..15]
|
|
# reverse for big endian
|
|
$Null = [array]::Reverse($TTLRaw)
|
|
$TTL = [BitConverter]::ToUInt32($TTLRaw, 0)
|
|
|
|
$Age = [BitConverter]::ToUInt32($DNSRecord, 20)
|
|
if($Age -ne 0) {
|
|
$TimeStamp = ((Get-Date -Year 1601 -Month 1 -Day 1 -Hour 0 -Minute 0 -Second 0).AddHours($age)).ToString()
|
|
}
|
|
else {
|
|
$TimeStamp = "[static]"
|
|
}
|
|
|
|
$DNSRecordObject = New-Object PSObject
|
|
|
|
if($RDataType -eq 1) {
|
|
$IP = "{0}.{1}.{2}.{3}" -f $DNSRecord[24], $DNSRecord[25], $DNSRecord[26], $DNSRecord[27]
|
|
$Data = $IP
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'A'
|
|
}
|
|
|
|
elseif($RDataType -eq 2) {
|
|
$NSName = Get-Name $DNSRecord[24..$DNSRecord.length]
|
|
$Data = $NSName
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'NS'
|
|
}
|
|
|
|
elseif($RDataType -eq 5) {
|
|
$Alias = Get-Name $DNSRecord[24..$DNSRecord.length]
|
|
$Data = $Alias
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'CNAME'
|
|
}
|
|
|
|
elseif($RDataType -eq 6) {
|
|
# TODO: how to implement properly? nested object?
|
|
$Data = $([System.Convert]::ToBase64String($DNSRecord[24..$DNSRecord.length]))
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'SOA'
|
|
}
|
|
|
|
elseif($RDataType -eq 12) {
|
|
$Ptr = Get-Name $DNSRecord[24..$DNSRecord.length]
|
|
$Data = $Ptr
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'PTR'
|
|
}
|
|
|
|
elseif($RDataType -eq 13) {
|
|
# TODO: how to implement properly? nested object?
|
|
$Data = $([System.Convert]::ToBase64String($DNSRecord[24..$DNSRecord.length]))
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'HINFO'
|
|
}
|
|
|
|
elseif($RDataType -eq 15) {
|
|
# TODO: how to implement properly? nested object?
|
|
$Data = $([System.Convert]::ToBase64String($DNSRecord[24..$DNSRecord.length]))
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'MX'
|
|
}
|
|
|
|
elseif($RDataType -eq 16) {
|
|
|
|
[string]$TXT = ""
|
|
[int]$SegmentLength = $DNSRecord[24]
|
|
$Index = 25
|
|
while ($SegmentLength-- -gt 0) {
|
|
$TXT += [char]$DNSRecord[$index++]
|
|
}
|
|
|
|
$Data = $TXT
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'TXT'
|
|
}
|
|
|
|
elseif($RDataType -eq 28) {
|
|
# TODO: how to implement properly? nested object?
|
|
$Data = $([System.Convert]::ToBase64String($DNSRecord[24..$DNSRecord.length]))
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'AAAA'
|
|
}
|
|
|
|
elseif($RDataType -eq 33) {
|
|
# TODO: how to implement properly? nested object?
|
|
$Data = $([System.Convert]::ToBase64String($DNSRecord[24..$DNSRecord.length]))
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'SRV'
|
|
}
|
|
|
|
else {
|
|
$Data = $([System.Convert]::ToBase64String($DNSRecord[24..$DNSRecord.length]))
|
|
$DNSRecordObject | Add-Member Noteproperty 'RecordType' 'UNKNOWN'
|
|
}
|
|
|
|
$DNSRecordObject | Add-Member Noteproperty 'UpdatedAtSerial' $UpdatedAtSerial
|
|
$DNSRecordObject | Add-Member Noteproperty 'TTL' $TTL
|
|
$DNSRecordObject | Add-Member Noteproperty 'Age' $Age
|
|
$DNSRecordObject | Add-Member Noteproperty 'TimeStamp' $TimeStamp
|
|
$DNSRecordObject | Add-Member Noteproperty 'Data' $Data
|
|
$DNSRecordObject
|
|
}
|
|
|
|
filter Get-DNSZone {
|
|
param(
|
|
[Parameter(Position=0, ValueFromPipeline=$True)]
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential,
|
|
|
|
[Switch]
|
|
$FullData
|
|
)
|
|
|
|
$DNSSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -PageSize $PageSize -Credential $Credential
|
|
$DNSSearcher.filter="(objectClass=dnsZone)"
|
|
|
|
if($DNSSearcher) {
|
|
$Results = $DNSSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
# convert/process the LDAP fields for each result
|
|
$Properties = Convert-LDAPProperty -Properties $_.Properties
|
|
$Properties | Add-Member NoteProperty 'ZoneName' $Properties.name
|
|
|
|
if ($FullData) {
|
|
$Properties
|
|
}
|
|
else {
|
|
$Properties | Select-Object ZoneName,distinguishedname,whencreated,whenchanged
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$DNSSearcher.dispose()
|
|
}
|
|
|
|
$DNSSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -PageSize $PageSize -Credential $Credential -ADSprefix "CN=MicrosoftDNS,DC=DomainDnsZones"
|
|
$DNSSearcher.filter="(objectClass=dnsZone)"
|
|
|
|
if($DNSSearcher) {
|
|
$Results = $DNSSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
# convert/process the LDAP fields for each result
|
|
$Properties = Convert-LDAPProperty -Properties $_.Properties
|
|
$Properties | Add-Member NoteProperty 'ZoneName' $Properties.name
|
|
|
|
if ($FullData) {
|
|
$Properties
|
|
}
|
|
else {
|
|
$Properties | Select-Object ZoneName,distinguishedname,whencreated,whenchanged
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$DNSSearcher.dispose()
|
|
}
|
|
}
|
|
|
|
filter Get-DNSRecord {
|
|
param(
|
|
[Parameter(Position=0, ValueFromPipelineByPropertyName=$True, Mandatory=$True)]
|
|
[String]
|
|
$ZoneName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
$DNSSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -PageSize $PageSize -Credential $Credential -ADSprefix "DC=$($ZoneName),CN=MicrosoftDNS,DC=DomainDnsZones"
|
|
$DNSSearcher.filter="(objectClass=dnsNode)"
|
|
|
|
if($DNSSearcher) {
|
|
$Results = $DNSSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
try {
|
|
# convert/process the LDAP fields for each result
|
|
$Properties = Convert-LDAPProperty -Properties $_.Properties | Select-Object name,distinguishedname,dnsrecord,whencreated,whenchanged
|
|
$Properties | Add-Member NoteProperty 'ZoneName' $ZoneName
|
|
|
|
# convert the record and extract the properties
|
|
if ($Properties.dnsrecord -is [System.DirectoryServices.ResultPropertyValueCollection]) {
|
|
# TODO: handle multiple nested records properly?
|
|
$Record = Convert-DNSRecord -DNSRecord $Properties.dnsrecord[0]
|
|
}
|
|
else {
|
|
$Record = Convert-DNSRecord -DNSRecord $Properties.dnsrecord
|
|
}
|
|
|
|
if($Record) {
|
|
$Record.psobject.properties | ForEach-Object {
|
|
$Properties | Add-Member NoteProperty $_.Name $_.Value
|
|
}
|
|
}
|
|
|
|
$Properties
|
|
}
|
|
catch {
|
|
#Write-Warning "ERROR: $_"
|
|
$Properties
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$DNSSearcher.dispose()
|
|
}
|
|
}
|
|
|
|
filter Get-ThisThingDomain {
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Domain,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
if($Credential) {
|
|
|
|
Write-Verbose "Using alternate credentials for Get-ThisThingDomain"
|
|
|
|
if(!$Domain) {
|
|
# if no domain is supplied, extract the logon domain from the PSCredential passed
|
|
$Domain = $Credential.GetNetworkCredential().Domain
|
|
Write-Verbose "Extracted domain '$Domain' from -Credential"
|
|
}
|
|
|
|
$DomainContext = New-Object System.DirectoryServices.ActiveDirectory.DirectoryContext('Domain', $Domain, $Credential.UserName, $Credential.GetNetworkCredential().Password)
|
|
|
|
try {
|
|
[System.DirectoryServices.ActiveDirectory.Domain]::GetDomain($DomainContext)
|
|
}
|
|
catch {
|
|
Write-Verbose "The specified domain does '$Domain' not exist, could not be contacted, there isn't an existing trust, or the specified credentials are invalid."
|
|
$Null
|
|
}
|
|
}
|
|
elseif($Domain) {
|
|
$DomainContext = New-Object System.DirectoryServices.ActiveDirectory.DirectoryContext('Domain', $Domain)
|
|
try {
|
|
[System.DirectoryServices.ActiveDirectory.Domain]::GetDomain($DomainContext)
|
|
}
|
|
catch {
|
|
Write-Verbose "The specified domain '$Domain' does not exist, could not be contacted, or there isn't an existing trust."
|
|
$Null
|
|
}
|
|
}
|
|
else {
|
|
[System.DirectoryServices.ActiveDirectory.Domain]::GetCurrentDomain()
|
|
}
|
|
}
|
|
|
|
filter Get-ThisThingForest {
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Forest,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
if($Credential) {
|
|
|
|
Write-Verbose "Using alternate credentials for Get-ThisThingForest"
|
|
|
|
if(!$Forest) {
|
|
# if no domain is supplied, extract the logon domain from the PSCredential passed
|
|
$Forest = $Credential.GetNetworkCredential().Domain
|
|
Write-Verbose "Extracted domain '$Forest' from -Credential"
|
|
}
|
|
|
|
$ForestContext = New-Object System.DirectoryServices.ActiveDirectory.DirectoryContext('Forest', $Forest, $Credential.UserName, $Credential.GetNetworkCredential().Password)
|
|
|
|
try {
|
|
$ForestObject = [System.DirectoryServices.ActiveDirectory.Forest]::GetForest($ForestContext)
|
|
}
|
|
catch {
|
|
Write-Verbose "The specified forest '$Forest' does not exist, could not be contacted, there isn't an existing trust, or the specified credentials are invalid."
|
|
$Null
|
|
}
|
|
}
|
|
elseif($Forest) {
|
|
$ForestContext = New-Object System.DirectoryServices.ActiveDirectory.DirectoryContext('Forest', $Forest)
|
|
try {
|
|
$ForestObject = [System.DirectoryServices.ActiveDirectory.Forest]::GetForest($ForestContext)
|
|
}
|
|
catch {
|
|
Write-Verbose "The specified forest '$Forest' does not exist, could not be contacted, or there isn't an existing trust."
|
|
return $Null
|
|
}
|
|
}
|
|
else {
|
|
# otherwise use the current forest
|
|
$ForestObject = [System.DirectoryServices.ActiveDirectory.Forest]::GetCurrentForest()
|
|
}
|
|
|
|
if($ForestObject) {
|
|
# get the SID of the forest root
|
|
$ForestSid = (New-Object System.Security.Principal.NTAccount($ForestObject.RootDomain,"krbtgt")).Translate([System.Security.Principal.SecurityIdentifier]).Value
|
|
$Parts = $ForestSid -Split "-"
|
|
$ForestSid = $Parts[0..$($Parts.length-2)] -join "-"
|
|
$ForestObject | Add-Member NoteProperty 'RootDomainSid' $ForestSid
|
|
$ForestObject
|
|
}
|
|
}
|
|
|
|
filter Get-ThisThingForestDomain {
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Forest,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
$ForestObject = Get-ThisThingForest -Forest $Forest -Credential $Credential
|
|
|
|
if($ForestObject) {
|
|
$ForestObject.Domains
|
|
}
|
|
}
|
|
|
|
filter Get-ThisThingForestCatalog {
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Forest,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
$ForestObject = Get-ThisThingForest -Forest $Forest -Credential $Credential
|
|
|
|
if($ForestObject) {
|
|
$ForestObject.FindAllGlobalCatalogs()
|
|
}
|
|
}
|
|
|
|
filter Get-ThisThingDomainController {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$LDAP,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
if($LDAP -or $DomainController) {
|
|
# filter string to return all domain controllers
|
|
Get-ThisThingComputer -Domain $Domain -DomainController $DomainController -Credential $Credential -FullData -Filter '(userAccountControl:1.2.840.113556.1.4.803:=8192)'
|
|
}
|
|
else {
|
|
$FoundDomain = Get-ThisThingDomain -Domain $Domain -Credential $Credential
|
|
if($FoundDomain) {
|
|
$Founddomain.DomainControllers
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingUser {
|
|
param(
|
|
[Parameter(Position=0, ValueFromPipeline=$True)]
|
|
[String]
|
|
$UserName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[Switch]
|
|
$SPN,
|
|
|
|
[Switch]
|
|
$AdminCount,
|
|
|
|
[Switch]
|
|
$Unconstrained,
|
|
|
|
[Switch]
|
|
$AllowDelegation,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
# so this isn't repeated if users are passed on the pipeline
|
|
$UserSearcher = Get-DomainSearcher -Domain $Domain -ADSpath $ADSpath -DomainController $DomainController -PageSize $PageSize -Credential $Credential
|
|
}
|
|
|
|
process {
|
|
if($UserSearcher) {
|
|
|
|
# if we're checking for unconstrained delegation
|
|
if($Unconstrained) {
|
|
Write-Verbose "Checking for unconstrained delegation"
|
|
$Filter += "(userAccountControl:1.2.840.113556.1.4.803:=524288)"
|
|
}
|
|
if($AllowDelegation) {
|
|
Write-Verbose "Checking for users who can be delegated"
|
|
# negation of "Accounts that are sensitive and not trusted for delegation"
|
|
$Filter += "(!(userAccountControl:1.2.840.113556.1.4.803:=1048574))"
|
|
}
|
|
if($AdminCount) {
|
|
Write-Verbose "Checking for adminCount=1"
|
|
$Filter += "(admincount=1)"
|
|
}
|
|
|
|
# check if we're using a username filter or not
|
|
if($UserName) {
|
|
# samAccountType=805306368 indicates user objects
|
|
$UserSearcher.filter="(&(samAccountType=805306368)(samAccountName=$UserName)$Filter)"
|
|
}
|
|
elseif($SPN) {
|
|
$UserSearcher.filter="(&(samAccountType=805306368)(servicePrincipalName=*)$Filter)"
|
|
}
|
|
else {
|
|
# filter is something like "(samAccountName=*blah*)" if specified
|
|
$UserSearcher.filter="(&(samAccountType=805306368)$Filter)"
|
|
}
|
|
|
|
$Results = $UserSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
# convert/process the LDAP fields for each result
|
|
$User = Convert-LDAPProperty -Properties $_.Properties
|
|
$User.PSObject.TypeNames.Add('PowerView.User')
|
|
$User
|
|
}
|
|
$Results.dispose()
|
|
$UserSearcher.dispose()
|
|
}
|
|
}
|
|
}
|
|
|
|
function Add-NetUser {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$UserName = 'backdoor',
|
|
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$Password = 'Password123!',
|
|
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$GroupName,
|
|
|
|
[ValidateNotNullOrEmpty()]
|
|
[Alias('HostName')]
|
|
[String]
|
|
$ComputerName = 'localhost',
|
|
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$Domain
|
|
)
|
|
|
|
if ($Domain) {
|
|
|
|
$DomainObject = Get-ThisThingDomain -Domain $Domain
|
|
if(-not $DomainObject) {
|
|
#Write-Warning "Error in grabbing $Domain object"
|
|
return $Null
|
|
}
|
|
|
|
# add the assembly we need
|
|
Add-Type -AssemblyName System.DirectoryServices.AccountManagement
|
|
|
|
# http://richardspowershellblog.wordpress.com/2008/05/25/system-directoryservices-accountmanagement/
|
|
# get the domain context
|
|
$Context = New-Object -TypeName System.DirectoryServices.AccountManagement.PrincipalContext -ArgumentList ([System.DirectoryServices.AccountManagement.ContextType]::Domain), $DomainObject
|
|
|
|
# create the user object
|
|
$User = New-Object -TypeName System.DirectoryServices.AccountManagement.UserPrincipal -ArgumentList $Context
|
|
|
|
# set user properties
|
|
$User.Name = $UserName
|
|
$User.SamAccountName = $UserName
|
|
$User.PasswordNotRequired = $False
|
|
$User.SetPassword($Password)
|
|
$User.Enabled = $True
|
|
|
|
Write-Verbose "Creating user $UserName to with password '$Password' in domain $Domain"
|
|
|
|
try {
|
|
# commit the user
|
|
$User.Save()
|
|
"[*] User $UserName successfully created in domain $Domain"
|
|
}
|
|
catch {
|
|
#Write-Warning '[!] User already exists!'
|
|
return
|
|
}
|
|
}
|
|
else {
|
|
|
|
Write-Verbose "Creating user $UserName to with password '$Password' on $ComputerName"
|
|
|
|
# if it's not a domain add, it's a local machine add
|
|
$ObjOu = [ADSI]"WinNT://$ComputerName"
|
|
$ObjUser = $ObjOu.Create('User', $UserName)
|
|
$ObjUser.SetPassword($Password)
|
|
|
|
# commit the changes to the local machine
|
|
try {
|
|
$Null = $ObjUser.SetInfo()
|
|
"[*] User $UserName successfully created on host $ComputerName"
|
|
}
|
|
catch {
|
|
#Write-Warning '[!] Account already exists!'
|
|
return
|
|
}
|
|
}
|
|
|
|
# if a group is specified, invoke Add-NetGroupUser and return its value
|
|
if ($GroupName) {
|
|
# if we're adding the user to a domain
|
|
if ($Domain) {
|
|
Add-NetGroupUser -UserName $UserName -GroupName $GroupName -Domain $Domain
|
|
"[*] User $UserName successfully added to group $GroupName in domain $Domain"
|
|
}
|
|
# otherwise, we're adding to a local group
|
|
else {
|
|
Add-NetGroupUser -UserName $UserName -GroupName $GroupName -ComputerName $ComputerName
|
|
"[*] User $UserName successfully added to group $GroupName on host $ComputerName"
|
|
}
|
|
}
|
|
}
|
|
|
|
function Add-NetGroupUser {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory = $True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$UserName,
|
|
|
|
[Parameter(Mandatory = $True)]
|
|
[ValidateNotNullOrEmpty()]
|
|
[String]
|
|
$GroupName,
|
|
|
|
[ValidateNotNullOrEmpty()]
|
|
[Alias('HostName')]
|
|
[String]
|
|
$ComputerName,
|
|
|
|
[String]
|
|
$Domain
|
|
)
|
|
|
|
# add the assembly if we need it
|
|
Add-Type -AssemblyName System.DirectoryServices.AccountManagement
|
|
|
|
# if we're adding to a remote host's local group, use the WinNT provider
|
|
if($ComputerName -and ($ComputerName -ne "localhost")) {
|
|
try {
|
|
Write-Verbose "Adding user $UserName to $GroupName on host $ComputerName"
|
|
([ADSI]"WinNT://$ComputerName/$GroupName,group").add("WinNT://$ComputerName/$UserName,user")
|
|
"[*] User $UserName successfully added to group $GroupName on $ComputerName"
|
|
}
|
|
catch {
|
|
Write-Warning "[!] Error adding user $UserName to group $GroupName on $ComputerName"
|
|
return
|
|
}
|
|
}
|
|
|
|
# otherwise it's a local machine or domain add
|
|
else {
|
|
try {
|
|
if ($Domain) {
|
|
Write-Verbose "Adding user $UserName to $GroupName on domain $Domain"
|
|
$CT = [System.DirectoryServices.AccountManagement.ContextType]::Domain
|
|
$DomainObject = Get-ThisThingDomain -Domain $Domain
|
|
if(-not $DomainObject) {
|
|
return $Null
|
|
}
|
|
# get the full principal context
|
|
$Context = New-Object -TypeName System.DirectoryServices.AccountManagement.PrincipalContext -ArgumentList $CT, $DomainObject
|
|
}
|
|
else {
|
|
# otherwise, get the local machine context
|
|
Write-Verbose "Adding user $UserName to $GroupName on localhost"
|
|
$Context = New-Object System.DirectoryServices.AccountManagement.PrincipalContext([System.DirectoryServices.AccountManagement.ContextType]::Machine, $Env:ComputerName)
|
|
}
|
|
|
|
# find the particular group
|
|
$Group = [System.DirectoryServices.AccountManagement.GroupPrincipal]::FindByIdentity($Context,$GroupName)
|
|
|
|
# add the particular user to the group
|
|
$Group.Members.add($Context, [System.DirectoryServices.AccountManagement.IdentityType]::SamAccountName, $UserName)
|
|
|
|
# commit the changes
|
|
$Group.Save()
|
|
}
|
|
catch {
|
|
Write-Warning "Error adding $UserName to $GroupName : $_"
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-UserProperty {
|
|
[CmdletBinding()]
|
|
param(
|
|
[String[]]
|
|
$Properties,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
if($Properties) {
|
|
# extract out the set of all properties for each object
|
|
$Properties = ,"name" + $Properties
|
|
Get-ThisThingUser -Domain $Domain -DomainController $DomainController -PageSize $PageSize -Credential $Credential | Select-Object -Property $Properties
|
|
}
|
|
else {
|
|
# extract out just the property names
|
|
Get-ThisThingUser -Domain $Domain -DomainController $DomainController -PageSize $PageSize -Credential $Credential | Select-Object -First 1 | Get-Member -MemberType *Property | Select-Object -Property 'Name'
|
|
}
|
|
}
|
|
|
|
filter Find-UserField {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0,ValueFromPipeline=$True)]
|
|
[String]
|
|
$SearchTerm = 'pass',
|
|
|
|
[String]
|
|
$SearchField = 'description',
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
Get-ThisThingUser -ADSpath $ADSpath -Domain $Domain -DomainController $DomainController -Credential $Credential -Filter "($SearchField=*$SearchTerm*)" -PageSize $PageSize | Select-Object samaccountname,$SearchField
|
|
}
|
|
|
|
filter Get-UserEvent {
|
|
Param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$ComputerName = $Env:ComputerName,
|
|
|
|
[String]
|
|
[ValidateSet("logon","tgt","all")]
|
|
$EventType = "logon",
|
|
|
|
[DateTime]
|
|
$DateStart = [DateTime]::Today.AddDays(-5),
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
if($EventType.ToLower() -like "logon") {
|
|
[Int32[]]$ID = @(4624)
|
|
}
|
|
elseif($EventType.ToLower() -like "tgt") {
|
|
[Int32[]]$ID = @(4768)
|
|
}
|
|
else {
|
|
[Int32[]]$ID = @(4624, 4768)
|
|
}
|
|
|
|
if($Credential) {
|
|
Write-Verbose "Using alternative credentials"
|
|
$Arguments = @{
|
|
'ComputerName' = $ComputerName;
|
|
'Credential' = $Credential;
|
|
'FilterHashTable' = @{ LogName = 'Security'; ID=$ID; StartTime=$DateStart};
|
|
'ErrorAction' = 'SilentlyContinue';
|
|
}
|
|
}
|
|
else {
|
|
$Arguments = @{
|
|
'ComputerName' = $ComputerName;
|
|
'FilterHashTable' = @{ LogName = 'Security'; ID=$ID; StartTime=$DateStart};
|
|
'ErrorAction' = 'SilentlyContinue';
|
|
}
|
|
}
|
|
|
|
# grab all events matching our filter for the specified host
|
|
Get-WinEvent @Arguments | ForEach-Object {
|
|
|
|
if($ID -contains 4624) {
|
|
# first parse and check the logon event type. This could be later adapted and tested for RDP logons (type 10)
|
|
if($_.message -match '(?s)(?<=Logon Type:).*?(?=(Impersonation Level:|New Logon:))') {
|
|
if($Matches) {
|
|
$LogonType = $Matches[0].trim()
|
|
$Matches = $Null
|
|
}
|
|
}
|
|
else {
|
|
$LogonType = ""
|
|
}
|
|
|
|
# interactive logons or domain logons
|
|
if (($LogonType -eq 2) -or ($LogonType -eq 3)) {
|
|
try {
|
|
# parse and store the account used and the address they came from
|
|
if($_.message -match '(?s)(?<=New Logon:).*?(?=Process Information:)') {
|
|
if($Matches) {
|
|
$UserName = $Matches[0].split("`n")[2].split(":")[1].trim()
|
|
$Domain = $Matches[0].split("`n")[3].split(":")[1].trim()
|
|
$Matches = $Null
|
|
}
|
|
}
|
|
if($_.message -match '(?s)(?<=Network Information:).*?(?=Source Port:)') {
|
|
if($Matches) {
|
|
$Address = $Matches[0].split("`n")[2].split(":")[1].trim()
|
|
$Matches = $Null
|
|
}
|
|
}
|
|
|
|
# only add if there was account information not for a machine or anonymous logon
|
|
if ($UserName -and (-not $UserName.endsWith('$')) -and ($UserName -ne 'ANONYMOUS LOGON')) {
|
|
$LogonEventProperties = @{
|
|
'Domain' = $Domain
|
|
'ComputerName' = $ComputerName
|
|
'Username' = $UserName
|
|
'Address' = $Address
|
|
'ID' = '4624'
|
|
'LogonType' = $LogonType
|
|
'Time' = $_.TimeCreated
|
|
}
|
|
New-Object -TypeName PSObject -Property $LogonEventProperties
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose "Error parsing event logs: $_"
|
|
}
|
|
}
|
|
}
|
|
if($ID -contains 4768) {
|
|
# the TGT event type
|
|
try {
|
|
if($_.message -match '(?s)(?<=Account Information:).*?(?=Service Information:)') {
|
|
if($Matches) {
|
|
$Username = $Matches[0].split("`n")[1].split(":")[1].trim()
|
|
$Domain = $Matches[0].split("`n")[2].split(":")[1].trim()
|
|
$Matches = $Null
|
|
}
|
|
}
|
|
|
|
if($_.message -match '(?s)(?<=Network Information:).*?(?=Additional Information:)') {
|
|
if($Matches) {
|
|
$Address = $Matches[0].split("`n")[1].split(":")[-1].trim()
|
|
$Matches = $Null
|
|
}
|
|
}
|
|
|
|
$LogonEventProperties = @{
|
|
'Domain' = $Domain
|
|
'ComputerName' = $ComputerName
|
|
'Username' = $UserName
|
|
'Address' = $Address
|
|
'ID' = '4768'
|
|
'LogonType' = ''
|
|
'Time' = $_.TimeCreated
|
|
}
|
|
|
|
New-Object -TypeName PSObject -Property $LogonEventProperties
|
|
}
|
|
catch {
|
|
Write-Verbose "Error parsing event logs: $_"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ObjectAcl {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipelineByPropertyName=$True)]
|
|
[String]
|
|
$SamAccountName,
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName=$True)]
|
|
[String]
|
|
$Name = "*",
|
|
|
|
[Parameter(ValueFromPipelineByPropertyName=$True)]
|
|
[String]
|
|
$DistinguishedName = "*",
|
|
|
|
[Switch]
|
|
$ResolveGUIDs,
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$ADSprefix,
|
|
|
|
[String]
|
|
[ValidateSet("All","ResetPassword","WriteMembers")]
|
|
$RightsFilter,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
begin {
|
|
$Searcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -ADSpath $ADSpath -ADSprefix $ADSprefix -PageSize $PageSize
|
|
|
|
# get a GUID -> name mapping
|
|
if($ResolveGUIDs) {
|
|
$GUIDs = Get-GUIDMap -Domain $Domain -DomainController $DomainController -PageSize $PageSize
|
|
}
|
|
}
|
|
|
|
process {
|
|
|
|
if ($Searcher) {
|
|
|
|
if($SamAccountName) {
|
|
$Searcher.filter="(&(samaccountname=$SamAccountName)(name=$Name)(distinguishedname=$DistinguishedName)$Filter)"
|
|
}
|
|
else {
|
|
$Searcher.filter="(&(name=$Name)(distinguishedname=$DistinguishedName)$Filter)"
|
|
}
|
|
|
|
try {
|
|
$Results = $Searcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
$Object = [adsi]($_.path)
|
|
|
|
if($Object.distinguishedname) {
|
|
$Access = $Object.PsBase.ObjectSecurity.access
|
|
$Access | ForEach-Object {
|
|
$_ | Add-Member NoteProperty 'ObjectDN' $Object.distinguishedname[0]
|
|
|
|
if($Object.objectsid[0]){
|
|
$S = (New-Object System.Security.Principal.SecurityIdentifier($Object.objectsid[0],0)).Value
|
|
}
|
|
else {
|
|
$S = $Null
|
|
}
|
|
|
|
$_ | Add-Member NoteProperty 'ObjectSID' $S
|
|
$_
|
|
}
|
|
}
|
|
} | ForEach-Object {
|
|
if($RightsFilter) {
|
|
$GuidFilter = Switch ($RightsFilter) {
|
|
"ResetPassword" { "00299570-246d-11d0-a768-00aa006e0529" }
|
|
"WriteMembers" { "bf9679c0-0de6-11d0-a285-00aa003049e2" }
|
|
Default { "00000000-0000-0000-0000-000000000000"}
|
|
}
|
|
if($_.ObjectType -eq $GuidFilter) { $_ }
|
|
}
|
|
else {
|
|
$_
|
|
}
|
|
} | ForEach-Object {
|
|
if($GUIDs) {
|
|
# if we're resolving GUIDs, map them them to the resolved hash table
|
|
$AclProperties = @{}
|
|
$_.psobject.properties | ForEach-Object {
|
|
if( ($_.Name -eq 'ObjectType') -or ($_.Name -eq 'InheritedObjectType') ) {
|
|
try {
|
|
$AclProperties[$_.Name] = $GUIDS[$_.Value.toString()]
|
|
}
|
|
catch {
|
|
$AclProperties[$_.Name] = $_.Value
|
|
}
|
|
}
|
|
else {
|
|
$AclProperties[$_.Name] = $_.Value
|
|
}
|
|
}
|
|
New-Object -TypeName PSObject -Property $AclProperties
|
|
}
|
|
else { $_ }
|
|
}
|
|
$Results.dispose()
|
|
$Searcher.dispose()
|
|
}
|
|
catch {
|
|
#Write-Warning $_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Add-ObjectAcl {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[String]
|
|
$TargetSamAccountName,
|
|
|
|
[String]
|
|
$TargetName = "*",
|
|
|
|
[Alias('DN')]
|
|
[String]
|
|
$TargetDistinguishedName = "*",
|
|
|
|
[String]
|
|
$TargetFilter,
|
|
|
|
[String]
|
|
$TargetADSpath,
|
|
|
|
[String]
|
|
$TargetADSprefix,
|
|
|
|
[String]
|
|
[ValidatePattern('^S-1-5-21-[0-9]+-[0-9]+-[0-9]+-[0-9]+')]
|
|
$PrincipalSID,
|
|
|
|
[String]
|
|
$PrincipalName,
|
|
|
|
[String]
|
|
$PrincipalSamAccountName,
|
|
|
|
[String]
|
|
[ValidateSet("All","ResetPassword","WriteMembers","DCSync")]
|
|
$Rights = "All",
|
|
|
|
[String]
|
|
$RightsGUID,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
begin {
|
|
$Searcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -ADSpath $TargetADSpath -ADSprefix $TargetADSprefix -PageSize $PageSize
|
|
|
|
if($PrincipalSID) {
|
|
$ResolvedPrincipalSID = $PrincipalSID
|
|
}
|
|
else {
|
|
$Principal = Get-ADObject -Domain $Domain -DomainController $DomainController -Name $PrincipalName -SamAccountName $PrincipalSamAccountName -PageSize $PageSize
|
|
|
|
if(!$Principal) {
|
|
throw "Error resolving principal"
|
|
}
|
|
$ResolvedPrincipalSID = $Principal.objectsid
|
|
}
|
|
if(!$ResolvedPrincipalSID) {
|
|
throw "Error resolving principal"
|
|
}
|
|
}
|
|
|
|
process {
|
|
|
|
if ($Searcher) {
|
|
|
|
if($TargetSamAccountName) {
|
|
$Searcher.filter="(&(samaccountname=$TargetSamAccountName)(name=$TargetName)(distinguishedname=$TargetDistinguishedName)$TargetFilter)"
|
|
}
|
|
else {
|
|
$Searcher.filter="(&(name=$TargetName)(distinguishedname=$TargetDistinguishedName)$TargetFilter)"
|
|
}
|
|
|
|
try {
|
|
$Results = $Searcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
|
|
# adapted from https://social.technet.microsoft.com/Forums/windowsserver/en-US/df3bfd33-c070-4a9c-be98-c4da6e591a0a/forum-faq-using-powershell-to-assign-permissions-on-active-directory-objects
|
|
|
|
$TargetDN = $_.Properties.distinguishedname
|
|
|
|
$Identity = [System.Security.Principal.IdentityReference] ([System.Security.Principal.SecurityIdentifier]$ResolvedPrincipalSID)
|
|
$InheritanceType = [System.DirectoryServices.ActiveDirectorySecurityInheritance] "None"
|
|
$ControlType = [System.Security.AccessControl.AccessControlType] "Allow"
|
|
$ACEs = @()
|
|
|
|
if($RightsGUID) {
|
|
$GUIDs = @($RightsGUID)
|
|
}
|
|
else {
|
|
$GUIDs = Switch ($Rights) {
|
|
# ResetPassword doesn't need to know the user's current password
|
|
"ResetPassword" { "00299570-246d-11d0-a768-00aa006e0529" }
|
|
# allows for the modification of group membership
|
|
"WriteMembers" { "bf9679c0-0de6-11d0-a285-00aa003049e2" }
|
|
# 'DS-Replication-Get-Changes' = 1131f6aa-9c07-11d1-f79f-00c04fc2dcd2
|
|
# 'DS-Replication-Get-Changes-All' = 1131f6ad-9c07-11d1-f79f-00c04fc2dcd2
|
|
# 'DS-Replication-Get-Changes-In-Filtered-Set' = 89e95b76-444d-4c62-991a-0facbeda640c
|
|
# when applied to a domain's ACL, allows for the use of DCSync
|
|
"DCSync" { "1131f6aa-9c07-11d1-f79f-00c04fc2dcd2", "1131f6ad-9c07-11d1-f79f-00c04fc2dcd2", "89e95b76-444d-4c62-991a-0facbeda640c"}
|
|
}
|
|
}
|
|
|
|
if($GUIDs) {
|
|
foreach($GUID in $GUIDs) {
|
|
$NewGUID = New-Object Guid $GUID
|
|
$ADRights = [System.DirectoryServices.ActiveDirectoryRights] "ExtendedRight"
|
|
$ACEs += New-Object System.DirectoryServices.ActiveDirectoryAccessRule $Identity,$ADRights,$ControlType,$NewGUID,$InheritanceType
|
|
}
|
|
}
|
|
else {
|
|
# deault to GenericAll rights
|
|
$ADRights = [System.DirectoryServices.ActiveDirectoryRights] "GenericAll"
|
|
$ACEs += New-Object System.DirectoryServices.ActiveDirectoryAccessRule $Identity,$ADRights,$ControlType,$InheritanceType
|
|
}
|
|
|
|
Write-Verbose "Granting principal $ResolvedPrincipalSID '$Rights' on $($_.Properties.distinguishedname)"
|
|
|
|
try {
|
|
# add all the new ACEs to the specified object
|
|
ForEach ($ACE in $ACEs) {
|
|
Write-Verbose "Granting principal $ResolvedPrincipalSID '$($ACE.ObjectType)' rights on $($_.Properties.distinguishedname)"
|
|
$Object = [adsi]($_.path)
|
|
$Object.PsBase.ObjectSecurity.AddAccessRule($ACE)
|
|
$Object.PsBase.commitchanges()
|
|
}
|
|
}
|
|
catch {
|
|
Write-Warning "Error granting principal $ResolvedPrincipalSID '$Rights' on $TargetDN : $_"
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$Searcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Warning "Error: $_"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Invoke-ACLScanner {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$SamAccountName,
|
|
|
|
[String]
|
|
$Name = "*",
|
|
|
|
[Alias('DN')]
|
|
[String]
|
|
$DistinguishedName = "*",
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$ADSprefix,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$ResolveGUIDs,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
# Get all domain ACLs with the appropriate parameters
|
|
Get-ObjectACL @PSBoundParameters | ForEach-Object {
|
|
# add in the translated SID for the object identity
|
|
$_ | Add-Member Noteproperty 'IdentitySID' ($_.IdentityReference.Translate([System.Security.Principal.SecurityIdentifier]).Value)
|
|
$_
|
|
} | Where-Object {
|
|
# check for any ACLs with SIDs > -1000
|
|
try {
|
|
# TODO: change this to a regex for speedup?
|
|
[int]($_.IdentitySid.split("-")[-1]) -ge 1000
|
|
}
|
|
catch {}
|
|
} | Where-Object {
|
|
# filter for modifiable rights
|
|
($_.ActiveDirectoryRights -eq "GenericAll") -or ($_.ActiveDirectoryRights -match "Write") -or ($_.ActiveDirectoryRights -match "Create") -or ($_.ActiveDirectoryRights -match "Delete") -or (($_.ActiveDirectoryRights -match "ExtendedRight") -and ($_.AccessControlType -eq "Allow"))
|
|
}
|
|
}
|
|
|
|
filter Get-GUIDMap {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
$GUIDs = @{'00000000-0000-0000-0000-000000000000' = 'All'}
|
|
|
|
$SchemaPath = (Get-ThisThingForest).schema.name
|
|
|
|
$SchemaSearcher = Get-DomainSearcher -ADSpath $SchemaPath -DomainController $DomainController -PageSize $PageSize
|
|
if($SchemaSearcher) {
|
|
$SchemaSearcher.filter = "(schemaIDGUID=*)"
|
|
try {
|
|
$Results = $SchemaSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
# convert the GUID
|
|
$GUIDs[(New-Object Guid (,$_.properties.schemaidguid[0])).Guid] = $_.properties.name[0]
|
|
}
|
|
$Results.dispose()
|
|
$SchemaSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Verbose "Error in building GUID map: $_"
|
|
}
|
|
}
|
|
|
|
$RightsSearcher = Get-DomainSearcher -ADSpath $SchemaPath.replace("Schema","Extended-Rights") -DomainController $DomainController -PageSize $PageSize -Credential $Credential
|
|
if ($RightsSearcher) {
|
|
$RightsSearcher.filter = "(objectClass=controlAccessRight)"
|
|
try {
|
|
$Results = $RightsSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
# convert the GUID
|
|
$GUIDs[$_.properties.rightsguid[0].toString()] = $_.properties.name[0]
|
|
}
|
|
$Results.dispose()
|
|
$RightsSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Verbose "Error in building GUID map: $_"
|
|
}
|
|
}
|
|
|
|
$GUIDs
|
|
}
|
|
|
|
function Get-ThisThingComputer {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[Alias('HostName')]
|
|
[String]
|
|
$ComputerName = '*',
|
|
|
|
[String]
|
|
$SPN,
|
|
|
|
[String]
|
|
$OperatingSystem,
|
|
|
|
[String]
|
|
$ServicePack,
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[Switch]
|
|
$Printers,
|
|
|
|
[Switch]
|
|
$Ping,
|
|
|
|
[Switch]
|
|
$FullData,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$SiteName,
|
|
|
|
[Switch]
|
|
$Unconstrained,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
# so this isn't repeated if multiple computer names are passed on the pipeline
|
|
$CompSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -ADSpath $ADSpath -PageSize $PageSize -Credential $Credential
|
|
}
|
|
|
|
process {
|
|
|
|
if ($CompSearcher) {
|
|
|
|
# if we're checking for unconstrained delegation
|
|
if($Unconstrained) {
|
|
Write-Verbose "Searching for computers with for unconstrained delegation"
|
|
$Filter += "(userAccountControl:1.2.840.113556.1.4.803:=524288)"
|
|
}
|
|
# set the filters for the seracher if it exists
|
|
if($Printers) {
|
|
Write-Verbose "Searching for printers"
|
|
# $CompSearcher.filter="(&(objectCategory=printQueue)$Filter)"
|
|
$Filter += "(objectCategory=printQueue)"
|
|
}
|
|
if($SPN) {
|
|
Write-Verbose "Searching for computers with SPN: $SPN"
|
|
$Filter += "(servicePrincipalName=$SPN)"
|
|
}
|
|
if($OperatingSystem) {
|
|
$Filter += "(operatingsystem=$OperatingSystem)"
|
|
}
|
|
if($ServicePack) {
|
|
$Filter += "(operatingsystemservicepack=$ServicePack)"
|
|
}
|
|
if($SiteName) {
|
|
$Filter += "(serverreferencebl=$SiteName)"
|
|
}
|
|
|
|
$CompFilter = "(&(sAMAccountType=805306369)(dnshostname=$ComputerName)$Filter)"
|
|
Write-Verbose "Get-ThisThingComputer filter : '$CompFilter'"
|
|
$CompSearcher.filter = $CompFilter
|
|
|
|
try {
|
|
$Results = $CompSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
$Up = $True
|
|
if($Ping) {
|
|
# TODO: how can these results be piped to ping for a speedup?
|
|
$Up = Test-Connection -Count 1 -Quiet -ComputerName $_.properties.dnshostname
|
|
}
|
|
if($Up) {
|
|
# return full data objects
|
|
if ($FullData) {
|
|
# convert/process the LDAP fields for each result
|
|
$Computer = Convert-LDAPProperty -Properties $_.Properties
|
|
$Computer.PSObject.TypeNames.Add('PowerView.Computer')
|
|
$Computer
|
|
}
|
|
else {
|
|
# otherwise we're just returning the DNS host name
|
|
$_.properties.dnshostname
|
|
}
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$CompSearcher.dispose()
|
|
}
|
|
catch {
|
|
#Write-Warning "Error: $_"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ADObject {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$SID,
|
|
|
|
[String]
|
|
$Name,
|
|
|
|
[String]
|
|
$SamAccountName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[Switch]
|
|
$ReturnRaw,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
process {
|
|
if($SID) {
|
|
# if a SID is passed, try to resolve it to a reachable domain name for the searcher
|
|
try {
|
|
$Name = Convert-SidToName $SID
|
|
if($Name) {
|
|
$Canonical = Convert-ADName -ObjectName $Name -InputType NT4 -OutputType Canonical
|
|
if($Canonical) {
|
|
$Domain = $Canonical.split("/")[0]
|
|
}
|
|
else {
|
|
#Write-Warning "Error resolving SID '$SID'"
|
|
return $Null
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
#Write-Warning "Error resolving SID '$SID' : $_"
|
|
return $Null
|
|
}
|
|
}
|
|
|
|
$ObjectSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
|
|
if($ObjectSearcher) {
|
|
if($SID) {
|
|
$ObjectSearcher.filter = "(&(objectsid=$SID)$Filter)"
|
|
}
|
|
elseif($Name) {
|
|
$ObjectSearcher.filter = "(&(name=$Name)$Filter)"
|
|
}
|
|
elseif($SamAccountName) {
|
|
$ObjectSearcher.filter = "(&(samAccountName=$SamAccountName)$Filter)"
|
|
}
|
|
|
|
$Results = $ObjectSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
if($ReturnRaw) {
|
|
$_
|
|
}
|
|
else {
|
|
# convert/process the LDAP fields for each result
|
|
Convert-LDAPProperty -Properties $_.Properties
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$ObjectSearcher.dispose()
|
|
}
|
|
}
|
|
}
|
|
|
|
function Set-ADObject {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[String]
|
|
$SID,
|
|
|
|
[String]
|
|
$Name,
|
|
|
|
[String]
|
|
$SamAccountName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[Parameter(Mandatory = $True)]
|
|
[String]
|
|
$PropertyName,
|
|
|
|
$PropertyValue,
|
|
|
|
[Int]
|
|
$PropertyXorValue,
|
|
|
|
[Switch]
|
|
$ClearValue,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
$Arguments = @{
|
|
'SID' = $SID
|
|
'Name' = $Name
|
|
'SamAccountName' = $SamAccountName
|
|
'Domain' = $Domain
|
|
'DomainController' = $DomainController
|
|
'Filter' = $Filter
|
|
'PageSize' = $PageSize
|
|
'Credential' = $Credential
|
|
}
|
|
# splat the appropriate arguments to Get-ADObject
|
|
$RawObject = Get-ADObject -ReturnRaw @Arguments
|
|
|
|
try {
|
|
# get the modifiable object for this search result
|
|
$Entry = $RawObject.GetDirectoryEntry()
|
|
|
|
if($ClearValue) {
|
|
Write-Verbose "Clearing value"
|
|
$Entry.$PropertyName.clear()
|
|
$Entry.commitchanges()
|
|
}
|
|
|
|
elseif($PropertyXorValue) {
|
|
$TypeName = $Entry.$PropertyName[0].GetType().name
|
|
|
|
# UAC value references- https://support.microsoft.com/en-us/kb/305144
|
|
$PropertyValue = $($Entry.$PropertyName) -bxor $PropertyXorValue
|
|
$Entry.$PropertyName = $PropertyValue -as $TypeName
|
|
$Entry.commitchanges()
|
|
}
|
|
|
|
else {
|
|
$Entry.put($PropertyName, $PropertyValue)
|
|
$Entry.setinfo()
|
|
}
|
|
}
|
|
catch {
|
|
Write-Warning "Error setting property $PropertyName to value '$PropertyValue' for object $($RawObject.Properties.samaccountname) : $_"
|
|
}
|
|
}
|
|
|
|
function Invoke-DowngradeAccount {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ParameterSetName = 'SamAccountName', Position=0, ValueFromPipeline=$True)]
|
|
[String]
|
|
$SamAccountName,
|
|
|
|
[Parameter(ParameterSetName = 'Name')]
|
|
[String]
|
|
$Name,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[Switch]
|
|
$Repair,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
process {
|
|
$Arguments = @{
|
|
'SamAccountName' = $SamAccountName
|
|
'Name' = $Name
|
|
'Domain' = $Domain
|
|
'DomainController' = $DomainController
|
|
'Filter' = $Filter
|
|
'Credential' = $Credential
|
|
}
|
|
|
|
# splat the appropriate arguments to Get-ADObject
|
|
$UACValues = Get-ADObject @Arguments | select useraccountcontrol | ConvertFrom-UACValue
|
|
|
|
if($Repair) {
|
|
|
|
if($UACValues.Keys -contains "ENCRYPTED_TEXT_PWD_ALLOWED") {
|
|
# if reversible encryption is set, unset it
|
|
Set-ADObject @Arguments -PropertyName useraccountcontrol -PropertyXorValue 128
|
|
}
|
|
|
|
# unset the forced password change
|
|
Set-ADObject @Arguments -PropertyName pwdlastset -PropertyValue -1
|
|
}
|
|
|
|
else {
|
|
|
|
if($UACValues.Keys -contains "DONT_EXPIRE_PASSWORD") {
|
|
# if the password is set to never expire, unset
|
|
Set-ADObject @Arguments -PropertyName useraccountcontrol -PropertyXorValue 65536
|
|
}
|
|
|
|
if($UACValues.Keys -notcontains "ENCRYPTED_TEXT_PWD_ALLOWED") {
|
|
# if reversible encryption is not set, set it
|
|
Set-ADObject @Arguments -PropertyName useraccountcontrol -PropertyXorValue 128
|
|
}
|
|
|
|
# force the password to be changed on next login
|
|
Set-ADObject @Arguments -PropertyName pwdlastset -PropertyValue 0
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ComputerProperty {
|
|
[CmdletBinding()]
|
|
param(
|
|
[String[]]
|
|
$Properties,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
if($Properties) {
|
|
# extract out the set of all properties for each object
|
|
$Properties = ,"name" + $Properties | Sort-Object -Unique
|
|
Get-ThisThingComputer -Domain $Domain -DomainController $DomainController -Credential $Credential -FullData -PageSize $PageSize | Select-Object -Property $Properties
|
|
}
|
|
else {
|
|
# extract out just the property names
|
|
Get-ThisThingComputer -Domain $Domain -DomainController $DomainController -Credential $Credential -FullData -PageSize $PageSize | Select-Object -first 1 | Get-Member -MemberType *Property | Select-Object -Property "Name"
|
|
}
|
|
}
|
|
|
|
function Find-ComputerField {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0,ValueFromPipeline=$True)]
|
|
[Alias('Term')]
|
|
[String]
|
|
$SearchTerm = 'pass',
|
|
|
|
[Alias('Field')]
|
|
[String]
|
|
$SearchField = 'description',
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
process {
|
|
Get-ThisThingComputer -ADSpath $ADSpath -Domain $Domain -DomainController $DomainController -Credential $Credential -FullData -Filter "($SearchField=*$SearchTerm*)" -PageSize $PageSize | Select-Object samaccountname,$SearchField
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingOU {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$OUName = '*',
|
|
|
|
[String]
|
|
$GUID,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[Switch]
|
|
$FullData,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
$OUSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
}
|
|
process {
|
|
if ($OUSearcher) {
|
|
if ($GUID) {
|
|
# if we're filtering for a GUID in .gplink
|
|
$OUSearcher.filter="(&(objectCategory=organizationalUnit)(name=$OUName)(gplink=*$GUID*))"
|
|
}
|
|
else {
|
|
$OUSearcher.filter="(&(objectCategory=organizationalUnit)(name=$OUName))"
|
|
}
|
|
|
|
try {
|
|
$Results = $OUSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
if ($FullData) {
|
|
# convert/process the LDAP fields for each result
|
|
$OU = Convert-LDAPProperty -Properties $_.Properties
|
|
$OU.PSObject.TypeNames.Add('PowerView.OU')
|
|
$OU
|
|
}
|
|
else {
|
|
# otherwise just returning the ADS paths of the OUs
|
|
$_.properties.adspath
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$OUSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Warning $_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingSite {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$SiteName = "*",
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$GUID,
|
|
|
|
[Switch]
|
|
$FullData,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
$SiteSearcher = Get-DomainSearcher -ADSpath $ADSpath -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSprefix "CN=Sites,CN=Configuration" -PageSize $PageSize
|
|
}
|
|
process {
|
|
if($SiteSearcher) {
|
|
|
|
if ($GUID) {
|
|
# if we're filtering for a GUID in .gplink
|
|
$SiteSearcher.filter="(&(objectCategory=site)(name=$SiteName)(gplink=*$GUID*))"
|
|
}
|
|
else {
|
|
$SiteSearcher.filter="(&(objectCategory=site)(name=$SiteName))"
|
|
}
|
|
|
|
try {
|
|
$Results = $SiteSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
if ($FullData) {
|
|
# convert/process the LDAP fields for each result
|
|
$Site = Convert-LDAPProperty -Properties $_.Properties
|
|
$Site.PSObject.TypeNames.Add('PowerView.Site')
|
|
$Site
|
|
}
|
|
else {
|
|
# otherwise just return the site name
|
|
$_.properties.name
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$SiteSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Verbose $_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingSubnet {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$SiteName = "*",
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$FullData,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
$SubnetSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -ADSprefix "CN=Subnets,CN=Sites,CN=Configuration" -PageSize $PageSize
|
|
}
|
|
|
|
process {
|
|
if($SubnetSearcher) {
|
|
|
|
$SubnetSearcher.filter="(&(objectCategory=subnet))"
|
|
|
|
try {
|
|
$Results = $SubnetSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
if ($FullData) {
|
|
# convert/process the LDAP fields for each result
|
|
Convert-LDAPProperty -Properties $_.Properties | Where-Object { $_.siteobject -match "CN=$SiteName" }
|
|
}
|
|
else {
|
|
# otherwise just return the subnet name and site name
|
|
if ( ($SiteName -and ($_.properties.siteobject -match "CN=$SiteName,")) -or ($SiteName -eq '*')) {
|
|
|
|
$SubnetProperties = @{
|
|
'Subnet' = $_.properties.name[0]
|
|
}
|
|
try {
|
|
$SubnetProperties['Site'] = ($_.properties.siteobject[0]).split(",")[0]
|
|
}
|
|
catch {
|
|
$SubnetProperties['Site'] = 'Error'
|
|
}
|
|
|
|
New-Object -TypeName PSObject -Property $SubnetProperties
|
|
}
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$SubnetSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Warning $_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-DomainSID {
|
|
|
|
|
|
param(
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController
|
|
)
|
|
|
|
$DCSID = Get-ThisThingComputer -Domain $Domain -DomainController $DomainController -FullData -Filter '(userAccountControl:1.2.840.113556.1.4.803:=8192)' | Select-Object -First 1 -ExpandProperty objectsid
|
|
if($DCSID) {
|
|
$DCSID.Substring(0, $DCSID.LastIndexOf('-'))
|
|
}
|
|
else {
|
|
Write-Verbose "Error extracting domain SID for $Domain"
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingGroup {
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$GroupName = '*',
|
|
|
|
[String]
|
|
$SID,
|
|
|
|
[String]
|
|
$UserName,
|
|
|
|
[String]
|
|
$Filter,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[Switch]
|
|
$AdminCount,
|
|
|
|
[Switch]
|
|
$FullData,
|
|
|
|
[Switch]
|
|
$RawSids,
|
|
|
|
[Switch]
|
|
$AllTypes,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
$GroupSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
if (!$AllTypes)
|
|
{
|
|
$Filter += "(groupType:1.2.840.113556.1.4.803:=2147483648)"
|
|
}
|
|
}
|
|
|
|
process {
|
|
if($GroupSearcher) {
|
|
|
|
if($AdminCount) {
|
|
Write-Verbose "Checking for adminCount=1"
|
|
$Filter += "(admincount=1)"
|
|
}
|
|
|
|
if ($UserName) {
|
|
# get the raw user object
|
|
$User = Get-ADObject -SamAccountName $UserName -Domain $Domain -DomainController $DomainController -Credential $Credential -ReturnRaw -PageSize $PageSize | Select-Object -First 1
|
|
|
|
if($User) {
|
|
# convert the user to a directory entry
|
|
$UserDirectoryEntry = $User.GetDirectoryEntry()
|
|
|
|
# cause the cache to calculate the token groups for the user
|
|
$UserDirectoryEntry.RefreshCache("tokenGroups")
|
|
|
|
$UserDirectoryEntry.TokenGroups | ForEach-Object {
|
|
# convert the token group sid
|
|
$GroupSid = (New-Object System.Security.Principal.SecurityIdentifier($_,0)).Value
|
|
|
|
# ignore the built in groups
|
|
if($GroupSid -notmatch '^S-1-5-32-.*') {
|
|
if($FullData) {
|
|
$Group = Get-ADObject -SID $GroupSid -PageSize $PageSize -Domain $Domain -DomainController $DomainController -Credential $Credential
|
|
$Group.PSObject.TypeNames.Add('PowerView.Group')
|
|
$Group
|
|
}
|
|
else {
|
|
if($RawSids) {
|
|
$GroupSid
|
|
}
|
|
else {
|
|
Convert-SidToName -SID $GroupSid
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
Write-Warning "UserName '$UserName' failed to resolve."
|
|
}
|
|
}
|
|
else {
|
|
if ($SID) {
|
|
$GroupSearcher.filter = "(&(objectCategory=group)(objectSID=$SID)$Filter)"
|
|
}
|
|
else {
|
|
$GroupSearcher.filter = "(&(objectCategory=group)(samaccountname=$GroupName)$Filter)"
|
|
}
|
|
|
|
$Results = $GroupSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
# if we're returning full data objects
|
|
if ($FullData) {
|
|
# convert/process the LDAP fields for each result
|
|
$Group = Convert-LDAPProperty -Properties $_.Properties
|
|
$Group.PSObject.TypeNames.Add('PowerView.Group')
|
|
$Group
|
|
}
|
|
else {
|
|
# otherwise we're just returning the group name
|
|
$_.properties.samaccountname
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$GroupSearcher.dispose()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingGroupMember {
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$GroupName,
|
|
|
|
[String]
|
|
$SID,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[Switch]
|
|
$FullData,
|
|
|
|
[Switch]
|
|
$Recurse,
|
|
|
|
[Switch]
|
|
$UseMatchingRule,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
if($DomainController) {
|
|
$TargetDomainController = $DomainController
|
|
}
|
|
else {
|
|
$TargetDomainController = ((Get-ThisThingDomain -Credential $Credential).PdcRoleOwner).Name
|
|
}
|
|
|
|
if($Domain) {
|
|
$TargetDomain = $Domain
|
|
}
|
|
else {
|
|
$TargetDomain = Get-ThisThingDomain -Credential $Credential | Select-Object -ExpandProperty name
|
|
}
|
|
|
|
# so this isn't repeated if users are passed on the pipeline
|
|
$GroupSearcher = Get-DomainSearcher -Domain $TargetDomain -DomainController $TargetDomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
}
|
|
|
|
process {
|
|
if ($GroupSearcher) {
|
|
if ($Recurse -and $UseMatchingRule) {
|
|
# resolve the group to a distinguishedname
|
|
if ($GroupName) {
|
|
$Group = Get-ThisThingGroup -AllTypes -GroupName $GroupName -Domain $TargetDomain -DomainController $TargetDomainController -Credential $Credential -FullData -PageSize $PageSize
|
|
}
|
|
elseif ($SID) {
|
|
$Group = Get-ThisThingGroup -AllTypes -SID $SID -Domain $TargetDomain -DomainController $TargetDomainController -Credential $Credential -FullData -PageSize $PageSize
|
|
}
|
|
else {
|
|
# default to domain admins
|
|
$SID = (Get-DomainSID -Domain $TargetDomain -DomainController $TargetDomainController) + "-512"
|
|
$Group = Get-ThisThingGroup -AllTypes -SID $SID -Domain $TargetDomain -DomainController $TargetDomainController -Credential $Credential -FullData -PageSize $PageSize
|
|
}
|
|
$GroupDN = $Group.distinguishedname
|
|
$GroupFoundName = $Group.samaccountname
|
|
|
|
if ($GroupDN) {
|
|
$GroupSearcher.filter = "(&(samAccountType=805306368)(memberof:1.2.840.113556.1.4.1941:=$GroupDN)$Filter)"
|
|
$GroupSearcher.PropertiesToLoad.AddRange(('distinguishedName','samaccounttype','lastlogon','lastlogontimestamp','dscorepropagationdata','objectsid','whencreated','badpasswordtime','accountexpires','iscriticalsystemobject','name','usnchanged','objectcategory','description','codepage','instancetype','countrycode','distinguishedname','cn','admincount','logonhours','objectclass','logoncount','usncreated','useraccountcontrol','objectguid','primarygroupid','lastlogoff','samaccountname','badpwdcount','whenchanged','memberof','pwdlastset','adspath'))
|
|
|
|
$Members = $GroupSearcher.FindAll()
|
|
$GroupFoundName = $GroupName
|
|
}
|
|
else {
|
|
Write-Error "Unable to find Group"
|
|
}
|
|
}
|
|
else {
|
|
if ($GroupName) {
|
|
$GroupSearcher.filter = "(&(objectCategory=group)(samaccountname=$GroupName)$Filter)"
|
|
}
|
|
elseif ($SID) {
|
|
$GroupSearcher.filter = "(&(objectCategory=group)(objectSID=$SID)$Filter)"
|
|
}
|
|
else {
|
|
# default to domain admins
|
|
$SID = (Get-DomainSID -Domain $TargetDomain -DomainController $TargetDomainController) + "-512"
|
|
$GroupSearcher.filter = "(&(objectCategory=group)(objectSID=$SID)$Filter)"
|
|
}
|
|
|
|
try {
|
|
$Result = $GroupSearcher.FindOne()
|
|
}
|
|
catch {
|
|
$Members = @()
|
|
}
|
|
|
|
$GroupFoundName = ''
|
|
|
|
if ($Result) {
|
|
$Members = $Result.properties.item("member")
|
|
|
|
if($Members.count -eq 0) {
|
|
|
|
$Finished = $False
|
|
$Bottom = 0
|
|
$Top = 0
|
|
|
|
while(!$Finished) {
|
|
$Top = $Bottom + 1499
|
|
$MemberRange="member;range=$Bottom-$Top"
|
|
$Bottom += 1500
|
|
|
|
$GroupSearcher.PropertiesToLoad.Clear()
|
|
[void]$GroupSearcher.PropertiesToLoad.Add("$MemberRange")
|
|
[void]$GroupSearcher.PropertiesToLoad.Add("samaccountname")
|
|
try {
|
|
$Result = $GroupSearcher.FindOne()
|
|
$RangedProperty = $Result.Properties.PropertyNames -like "member;range=*"
|
|
$Members += $Result.Properties.item($RangedProperty)
|
|
$GroupFoundName = $Result.properties.item("samaccountname")[0]
|
|
|
|
if ($Members.count -eq 0) {
|
|
$Finished = $True
|
|
}
|
|
}
|
|
catch [System.Management.Automation.MethodInvocationException] {
|
|
$Finished = $True
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
$GroupFoundName = $Result.properties.item("samaccountname")[0]
|
|
$Members += $Result.Properties.item($RangedProperty)
|
|
}
|
|
}
|
|
$GroupSearcher.dispose()
|
|
}
|
|
|
|
$Members | Where-Object {$_} | ForEach-Object {
|
|
# if we're doing the LDAP_MATCHING_RULE_IN_CHAIN recursion
|
|
if ($Recurse -and $UseMatchingRule) {
|
|
$Properties = $_.Properties
|
|
}
|
|
else {
|
|
if($TargetDomainController) {
|
|
$Result = [adsi]"LDAP://$TargetDomainController/$_"
|
|
}
|
|
else {
|
|
$Result = [adsi]"LDAP://$_"
|
|
}
|
|
if($Result){
|
|
$Properties = $Result.Properties
|
|
}
|
|
}
|
|
|
|
if($Properties) {
|
|
|
|
$IsGroup = @('268435456','268435457','536870912','536870913') -contains $Properties.samaccounttype
|
|
|
|
if ($FullData) {
|
|
$GroupMember = Convert-LDAPProperty -Properties $Properties
|
|
}
|
|
else {
|
|
$GroupMember = New-Object PSObject
|
|
}
|
|
|
|
$GroupMember | Add-Member Noteproperty 'GroupDomain' $TargetDomain
|
|
$GroupMember | Add-Member Noteproperty 'GroupName' $GroupFoundName
|
|
|
|
if($Properties.objectSid) {
|
|
$MemberSID = ((New-Object System.Security.Principal.SecurityIdentifier $Properties.objectSid[0],0).Value)
|
|
}
|
|
else {
|
|
$MemberSID = $Null
|
|
}
|
|
|
|
try {
|
|
$MemberDN = $Properties.distinguishedname[0]
|
|
|
|
if (($MemberDN -match 'ForeignSecurityPrincipals') -and ($MemberDN -match 'S-1-5-21')) {
|
|
try {
|
|
if(-not $MemberSID) {
|
|
$MemberSID = $Properties.cn[0]
|
|
}
|
|
$MemberSimpleName = Convert-SidToName -SID $MemberSID | Convert-ADName -InputType 'NT4' -OutputType 'Simple'
|
|
if($MemberSimpleName) {
|
|
$MemberDomain = $MemberSimpleName.Split('@')[1]
|
|
}
|
|
else {
|
|
Write-Warning "Error converting $MemberDN"
|
|
$MemberDomain = $Null
|
|
}
|
|
}
|
|
catch {
|
|
Write-Warning "Error converting $MemberDN"
|
|
$MemberDomain = $Null
|
|
}
|
|
}
|
|
else {
|
|
# extract the FQDN from the Distinguished Name
|
|
$MemberDomain = $MemberDN.subString($MemberDN.IndexOf("DC=")) -replace 'DC=','' -replace ',','.'
|
|
}
|
|
}
|
|
catch {
|
|
$MemberDN = $Null
|
|
$MemberDomain = $Null
|
|
}
|
|
|
|
if ($Properties.samaccountname) {
|
|
# forest users have the samAccountName set
|
|
$MemberName = $Properties.samaccountname[0]
|
|
}
|
|
else {
|
|
# external trust users have a SID, so convert it
|
|
try {
|
|
$MemberName = Convert-SidToName $Properties.cn[0]
|
|
}
|
|
catch {
|
|
# if there's a problem contacting the domain to resolve the SID
|
|
$MemberName = $Properties.cn
|
|
}
|
|
}
|
|
|
|
$GroupMember | Add-Member Noteproperty 'MemberDomain' $MemberDomain
|
|
$GroupMember | Add-Member Noteproperty 'MemberName' $MemberName
|
|
$GroupMember | Add-Member Noteproperty 'MemberSID' $MemberSID
|
|
$GroupMember | Add-Member Noteproperty 'IsGroup' $IsGroup
|
|
$GroupMember | Add-Member Noteproperty 'MemberDN' $MemberDN
|
|
$GroupMember.PSObject.TypeNames.Add('PowerView.GroupMember')
|
|
$GroupMember
|
|
|
|
# if we're doing manual recursion
|
|
if ($Recurse -and !$UseMatchingRule -and $IsGroup -and $MemberName) {
|
|
if($FullData) {
|
|
Get-ThisThingGroupMember -FullData -Domain $MemberDomain -DomainController $TargetDomainController -Credential $Credential -GroupName $MemberName -Recurse -PageSize $PageSize
|
|
}
|
|
else {
|
|
Get-ThisThingGroupMember -Domain $MemberDomain -DomainController $TargetDomainController -Credential $Credential -GroupName $MemberName -Recurse -PageSize $PageSize
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingFileServer {
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String[]]
|
|
$TargetUsers,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
function SplitPath {
|
|
# short internal helper to split UNC server paths
|
|
param([String]$Path)
|
|
|
|
if ($Path -and ($Path.split("\\").Count -ge 3)) {
|
|
$Temp = $Path.split("\\")[2]
|
|
if($Temp -and ($Temp -ne '')) {
|
|
$Temp
|
|
}
|
|
}
|
|
}
|
|
$filter = "(!(userAccountControl:1.2.840.113556.1.4.803:=2))(|(scriptpath=*)(homedirectory=*)(profilepath=*))"
|
|
Get-ThisThingUser -Domain $Domain -DomainController $DomainController -Credential $Credential -PageSize $PageSize -Filter $filter | Where-Object {$_} | Where-Object {
|
|
# filter for any target users
|
|
if($TargetUsers) {
|
|
$TargetUsers -Match $_.samAccountName
|
|
}
|
|
else { $True }
|
|
} | ForEach-Object {
|
|
# split out every potential file server path
|
|
if($_.homedirectory) {
|
|
SplitPath($_.homedirectory)
|
|
}
|
|
if($_.scriptpath) {
|
|
SplitPath($_.scriptpath)
|
|
}
|
|
if($_.profilepath) {
|
|
SplitPath($_.profilepath)
|
|
}
|
|
|
|
} | Where-Object {$_} | Sort-Object -Unique
|
|
}
|
|
|
|
function Get-DFSshare {
|
|
[CmdletBinding()]
|
|
param(
|
|
[String]
|
|
[ValidateSet("All","V1","1","V2","2")]
|
|
$Version = "All",
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
function Parse-Pkt {
|
|
[CmdletBinding()]
|
|
param(
|
|
[byte[]]
|
|
$Pkt
|
|
)
|
|
|
|
$bin = $Pkt
|
|
$blob_version = [bitconverter]::ToUInt32($bin[0..3],0)
|
|
$blob_element_count = [bitconverter]::ToUInt32($bin[4..7],0)
|
|
$offset = 8
|
|
#https://msdn.microsoft.com/en-us/library/cc227147.aspx
|
|
$object_list = @()
|
|
for($i=1; $i -le $blob_element_count; $i++){
|
|
$blob_name_size_start = $offset
|
|
$blob_name_size_end = $offset + 1
|
|
$blob_name_size = [bitconverter]::ToUInt16($bin[$blob_name_size_start..$blob_name_size_end],0)
|
|
|
|
$blob_name_start = $blob_name_size_end + 1
|
|
$blob_name_end = $blob_name_start + $blob_name_size - 1
|
|
$blob_name = [System.Text.Encoding]::Unicode.GetString($bin[$blob_name_start..$blob_name_end])
|
|
|
|
$blob_data_size_start = $blob_name_end + 1
|
|
$blob_data_size_end = $blob_data_size_start + 3
|
|
$blob_data_size = [bitconverter]::ToUInt32($bin[$blob_data_size_start..$blob_data_size_end],0)
|
|
|
|
$blob_data_start = $blob_data_size_end + 1
|
|
$blob_data_end = $blob_data_start + $blob_data_size - 1
|
|
$blob_data = $bin[$blob_data_start..$blob_data_end]
|
|
switch -wildcard ($blob_name) {
|
|
"\siteroot" { }
|
|
"\domainroot*" {
|
|
# Parse DFSNamespaceRootOrLinkBlob object. Starts with variable length DFSRootOrLinkIDBlob which we parse first...
|
|
# DFSRootOrLinkIDBlob
|
|
$root_or_link_guid_start = 0
|
|
$root_or_link_guid_end = 15
|
|
$root_or_link_guid = [byte[]]$blob_data[$root_or_link_guid_start..$root_or_link_guid_end]
|
|
$guid = New-Object Guid(,$root_or_link_guid) # should match $guid_str
|
|
$prefix_size_start = $root_or_link_guid_end + 1
|
|
$prefix_size_end = $prefix_size_start + 1
|
|
$prefix_size = [bitconverter]::ToUInt16($blob_data[$prefix_size_start..$prefix_size_end],0)
|
|
$prefix_start = $prefix_size_end + 1
|
|
$prefix_end = $prefix_start + $prefix_size - 1
|
|
$prefix = [System.Text.Encoding]::Unicode.GetString($blob_data[$prefix_start..$prefix_end])
|
|
|
|
$short_prefix_size_start = $prefix_end + 1
|
|
$short_prefix_size_end = $short_prefix_size_start + 1
|
|
$short_prefix_size = [bitconverter]::ToUInt16($blob_data[$short_prefix_size_start..$short_prefix_size_end],0)
|
|
$short_prefix_start = $short_prefix_size_end + 1
|
|
$short_prefix_end = $short_prefix_start + $short_prefix_size - 1
|
|
$short_prefix = [System.Text.Encoding]::Unicode.GetString($blob_data[$short_prefix_start..$short_prefix_end])
|
|
|
|
$type_start = $short_prefix_end + 1
|
|
$type_end = $type_start + 3
|
|
$type = [bitconverter]::ToUInt32($blob_data[$type_start..$type_end],0)
|
|
|
|
$state_start = $type_end + 1
|
|
$state_end = $state_start + 3
|
|
$state = [bitconverter]::ToUInt32($blob_data[$state_start..$state_end],0)
|
|
|
|
$comment_size_start = $state_end + 1
|
|
$comment_size_end = $comment_size_start + 1
|
|
$comment_size = [bitconverter]::ToUInt16($blob_data[$comment_size_start..$comment_size_end],0)
|
|
$comment_start = $comment_size_end + 1
|
|
$comment_end = $comment_start + $comment_size - 1
|
|
if ($comment_size -gt 0) {
|
|
$comment = [System.Text.Encoding]::Unicode.GetString($blob_data[$comment_start..$comment_end])
|
|
}
|
|
$prefix_timestamp_start = $comment_end + 1
|
|
$prefix_timestamp_end = $prefix_timestamp_start + 7
|
|
# https://msdn.microsoft.com/en-us/library/cc230324.aspx FILETIME
|
|
$prefix_timestamp = $blob_data[$prefix_timestamp_start..$prefix_timestamp_end] #dword lowDateTime #dword highdatetime
|
|
$state_timestamp_start = $prefix_timestamp_end + 1
|
|
$state_timestamp_end = $state_timestamp_start + 7
|
|
$state_timestamp = $blob_data[$state_timestamp_start..$state_timestamp_end]
|
|
$comment_timestamp_start = $state_timestamp_end + 1
|
|
$comment_timestamp_end = $comment_timestamp_start + 7
|
|
$comment_timestamp = $blob_data[$comment_timestamp_start..$comment_timestamp_end]
|
|
$version_start = $comment_timestamp_end + 1
|
|
$version_end = $version_start + 3
|
|
$version = [bitconverter]::ToUInt32($blob_data[$version_start..$version_end],0)
|
|
|
|
# Parse rest of DFSNamespaceRootOrLinkBlob here
|
|
$dfs_targetlist_blob_size_start = $version_end + 1
|
|
$dfs_targetlist_blob_size_end = $dfs_targetlist_blob_size_start + 3
|
|
$dfs_targetlist_blob_size = [bitconverter]::ToUInt32($blob_data[$dfs_targetlist_blob_size_start..$dfs_targetlist_blob_size_end],0)
|
|
|
|
$dfs_targetlist_blob_start = $dfs_targetlist_blob_size_end + 1
|
|
$dfs_targetlist_blob_end = $dfs_targetlist_blob_start + $dfs_targetlist_blob_size - 1
|
|
$dfs_targetlist_blob = $blob_data[$dfs_targetlist_blob_start..$dfs_targetlist_blob_end]
|
|
$reserved_blob_size_start = $dfs_targetlist_blob_end + 1
|
|
$reserved_blob_size_end = $reserved_blob_size_start + 3
|
|
$reserved_blob_size = [bitconverter]::ToUInt32($blob_data[$reserved_blob_size_start..$reserved_blob_size_end],0)
|
|
|
|
$reserved_blob_start = $reserved_blob_size_end + 1
|
|
$reserved_blob_end = $reserved_blob_start + $reserved_blob_size - 1
|
|
$reserved_blob = $blob_data[$reserved_blob_start..$reserved_blob_end]
|
|
$referral_ttl_start = $reserved_blob_end + 1
|
|
$referral_ttl_end = $referral_ttl_start + 3
|
|
$referral_ttl = [bitconverter]::ToUInt32($blob_data[$referral_ttl_start..$referral_ttl_end],0)
|
|
|
|
#Parse DFSTargetListBlob
|
|
$target_count_start = 0
|
|
$target_count_end = $target_count_start + 3
|
|
$target_count = [bitconverter]::ToUInt32($dfs_targetlist_blob[$target_count_start..$target_count_end],0)
|
|
$t_offset = $target_count_end + 1
|
|
|
|
for($j=1; $j -le $target_count; $j++){
|
|
$target_entry_size_start = $t_offset
|
|
$target_entry_size_end = $target_entry_size_start + 3
|
|
$target_entry_size = [bitconverter]::ToUInt32($dfs_targetlist_blob[$target_entry_size_start..$target_entry_size_end],0)
|
|
$target_time_stamp_start = $target_entry_size_end + 1
|
|
$target_time_stamp_end = $target_time_stamp_start + 7
|
|
# FILETIME again or special if priority rank and priority class 0
|
|
$target_time_stamp = $dfs_targetlist_blob[$target_time_stamp_start..$target_time_stamp_end]
|
|
$target_state_start = $target_time_stamp_end + 1
|
|
$target_state_end = $target_state_start + 3
|
|
$target_state = [bitconverter]::ToUInt32($dfs_targetlist_blob[$target_state_start..$target_state_end],0)
|
|
|
|
$target_type_start = $target_state_end + 1
|
|
$target_type_end = $target_type_start + 3
|
|
$target_type = [bitconverter]::ToUInt32($dfs_targetlist_blob[$target_type_start..$target_type_end],0)
|
|
|
|
$server_name_size_start = $target_type_end + 1
|
|
$server_name_size_end = $server_name_size_start + 1
|
|
$server_name_size = [bitconverter]::ToUInt16($dfs_targetlist_blob[$server_name_size_start..$server_name_size_end],0)
|
|
|
|
$server_name_start = $server_name_size_end + 1
|
|
$server_name_end = $server_name_start + $server_name_size - 1
|
|
$server_name = [System.Text.Encoding]::Unicode.GetString($dfs_targetlist_blob[$server_name_start..$server_name_end])
|
|
|
|
$share_name_size_start = $server_name_end + 1
|
|
$share_name_size_end = $share_name_size_start + 1
|
|
$share_name_size = [bitconverter]::ToUInt16($dfs_targetlist_blob[$share_name_size_start..$share_name_size_end],0)
|
|
$share_name_start = $share_name_size_end + 1
|
|
$share_name_end = $share_name_start + $share_name_size - 1
|
|
$share_name = [System.Text.Encoding]::Unicode.GetString($dfs_targetlist_blob[$share_name_start..$share_name_end])
|
|
|
|
$target_list += "\\$server_name\$share_name"
|
|
$t_offset = $share_name_end + 1
|
|
}
|
|
}
|
|
}
|
|
$offset = $blob_data_end + 1
|
|
$dfs_pkt_properties = @{
|
|
'Name' = $blob_name
|
|
'Prefix' = $prefix
|
|
'TargetList' = $target_list
|
|
}
|
|
$object_list += New-Object -TypeName PSObject -Property $dfs_pkt_properties
|
|
$prefix = $null
|
|
$blob_name = $null
|
|
$target_list = $null
|
|
}
|
|
|
|
$servers = @()
|
|
$object_list | ForEach-Object {
|
|
if ($_.TargetList) {
|
|
$_.TargetList | ForEach-Object {
|
|
$servers += $_.split("\")[2]
|
|
}
|
|
}
|
|
}
|
|
|
|
$servers
|
|
}
|
|
|
|
function Get-DFSshareV1 {
|
|
[CmdletBinding()]
|
|
param(
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
$DFSsearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
|
|
if($DFSsearcher) {
|
|
$DFSshares = @()
|
|
$DFSsearcher.filter = "(&(objectClass=fTDfs))"
|
|
|
|
try {
|
|
$Results = $DFSSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
$Properties = $_.Properties
|
|
$RemoteNames = $Properties.remoteservername
|
|
$Pkt = $Properties.pkt
|
|
|
|
$DFSshares += $RemoteNames | ForEach-Object {
|
|
try {
|
|
if ( $_.Contains('\') ) {
|
|
New-Object -TypeName PSObject -Property @{'Name'=$Properties.name[0];'RemoteServerName'=$_.split("\")[2]}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose "Error in parsing DFS share : $_"
|
|
}
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$DFSSearcher.dispose()
|
|
|
|
if($pkt -and $pkt[0]) {
|
|
Parse-Pkt $pkt[0] | ForEach-Object {
|
|
# If a folder doesn't have a redirection it will
|
|
# have a target like
|
|
# \\null\TestNameSpace\folder\.DFSFolderLink so we
|
|
# do actually want to match on "null" rather than
|
|
# $null
|
|
if ($_ -ne "null") {
|
|
New-Object -TypeName PSObject -Property @{'Name'=$Properties.name[0];'RemoteServerName'=$_}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Warning "Get-DFSshareV1 error : $_"
|
|
}
|
|
$DFSshares | Sort-Object -Property "RemoteServerName"
|
|
}
|
|
}
|
|
|
|
function Get-DFSshareV2 {
|
|
[CmdletBinding()]
|
|
param(
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
$DFSsearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
|
|
if($DFSsearcher) {
|
|
$DFSshares = @()
|
|
$DFSsearcher.filter = "(&(objectClass=msDFS-Linkv2))"
|
|
$DFSSearcher.PropertiesToLoad.AddRange(('msdfs-linkpathv2','msDFS-TargetListv2'))
|
|
|
|
try {
|
|
$Results = $DFSSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
$Properties = $_.Properties
|
|
$target_list = $Properties.'msdfs-targetlistv2'[0]
|
|
$xml = [xml][System.Text.Encoding]::Unicode.GetString($target_list[2..($target_list.Length-1)])
|
|
$DFSshares += $xml.targets.ChildNodes | ForEach-Object {
|
|
try {
|
|
$Target = $_.InnerText
|
|
if ( $Target.Contains('\') ) {
|
|
$DFSroot = $Target.split("\")[3]
|
|
$ShareName = $Properties.'msdfs-linkpathv2'[0]
|
|
New-Object -TypeName PSObject -Property @{'Name'="$DFSroot$ShareName";'RemoteServerName'=$Target.split("\")[2]}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose "Error in parsing target : $_"
|
|
}
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$DFSSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Warning "Get-DFSshareV2 error : $_"
|
|
}
|
|
$DFSshares | Sort-Object -Unique -Property "RemoteServerName"
|
|
}
|
|
}
|
|
|
|
$DFSshares = @()
|
|
|
|
if ( ($Version -eq "all") -or ($Version.endsWith("1")) ) {
|
|
$DFSshares += Get-DFSshareV1 -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
}
|
|
if ( ($Version -eq "all") -or ($Version.endsWith("2")) ) {
|
|
$DFSshares += Get-DFSshareV2 -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
}
|
|
|
|
$DFSshares | Sort-Object -Property ("RemoteServerName","Name") -Unique
|
|
}
|
|
|
|
filter Get-GptTmpl {
|
|
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
[String]
|
|
$GptTmplPath,
|
|
|
|
[Switch]
|
|
$UsePSDrive
|
|
)
|
|
|
|
if($UsePSDrive) {
|
|
# if we're PSDrives, create a temporary mount point
|
|
$Parts = $GptTmplPath.split('\')
|
|
$FolderPath = $Parts[0..($Parts.length-2)] -join '\'
|
|
$FilePath = $Parts[-1]
|
|
$RandDrive = ("abcdefghijklmnopqrstuvwxyz".ToCharArray() | Get-Random -Count 7) -join ''
|
|
|
|
Write-Verbose "Mounting path $GptTmplPath using a temp PSDrive at $RandDrive"
|
|
|
|
try {
|
|
$Null = New-PSDrive -Name $RandDrive -PSProvider FileSystem -Root $FolderPath -ErrorAction Stop
|
|
}
|
|
catch {
|
|
Write-Verbose "Error mounting path $GptTmplPath : $_"
|
|
return $Null
|
|
}
|
|
|
|
# so we can cd/dir the new drive
|
|
$TargetGptTmplPath = $RandDrive + ":\" + $FilePath
|
|
}
|
|
else {
|
|
$TargetGptTmplPath = $GptTmplPath
|
|
}
|
|
|
|
Write-Verbose "GptTmplPath: $GptTmplPath"
|
|
|
|
try {
|
|
Write-Verbose "Parsing $TargetGptTmplPath"
|
|
$TargetGptTmplPath | Get-IniContent -ErrorAction SilentlyContinue
|
|
}
|
|
catch {
|
|
Write-Verbose "Error parsing $TargetGptTmplPath : $_"
|
|
}
|
|
|
|
if($UsePSDrive -and $RandDrive) {
|
|
Write-Verbose "Removing temp PSDrive $RandDrive"
|
|
Get-PSDrive -Name $RandDrive -ErrorAction SilentlyContinue | Remove-PSDrive -Force
|
|
}
|
|
}
|
|
|
|
filter Get-GroupsXML {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(Mandatory=$True, ValueFromPipeline=$True)]
|
|
[String]
|
|
$GroupsXMLPath,
|
|
|
|
[Switch]
|
|
$UsePSDrive
|
|
)
|
|
|
|
if($UsePSDrive) {
|
|
# if we're PSDrives, create a temporary mount point
|
|
$Parts = $GroupsXMLPath.split('\')
|
|
$FolderPath = $Parts[0..($Parts.length-2)] -join '\'
|
|
$FilePath = $Parts[-1]
|
|
$RandDrive = ("abcdefghijklmnopqrstuvwxyz".ToCharArray() | Get-Random -Count 7) -join ''
|
|
|
|
Write-Verbose "Mounting path $GroupsXMLPath using a temp PSDrive at $RandDrive"
|
|
|
|
try {
|
|
$Null = New-PSDrive -Name $RandDrive -PSProvider FileSystem -Root $FolderPath -ErrorAction Stop
|
|
}
|
|
catch {
|
|
Write-Verbose "Error mounting path $GroupsXMLPath : $_"
|
|
return $Null
|
|
}
|
|
|
|
# so we can cd/dir the new drive
|
|
$TargetGroupsXMLPath = $RandDrive + ":\" + $FilePath
|
|
}
|
|
else {
|
|
$TargetGroupsXMLPath = $GroupsXMLPath
|
|
}
|
|
|
|
try {
|
|
[XML]$GroupsXMLcontent = Get-Content $TargetGroupsXMLPath -ErrorAction Stop
|
|
|
|
# process all group properties in the XML
|
|
$GroupsXMLcontent | Select-Xml "/Groups/Group" | Select-Object -ExpandProperty node | ForEach-Object {
|
|
|
|
$Groupname = $_.Properties.groupName
|
|
|
|
# extract the localgroup sid for memberof
|
|
$GroupSID = $_.Properties.groupSid
|
|
if(-not $GroupSID) {
|
|
if($Groupname -match 'Administrators') {
|
|
$GroupSID = 'S-1-5-32-544'
|
|
}
|
|
elseif($Groupname -match 'Remote Desktop') {
|
|
$GroupSID = 'S-1-5-32-555'
|
|
}
|
|
elseif($Groupname -match 'Guests') {
|
|
$GroupSID = 'S-1-5-32-546'
|
|
}
|
|
else {
|
|
$GroupSID = Convert-NameToSid -ObjectName $Groupname | Select-Object -ExpandProperty SID
|
|
}
|
|
}
|
|
|
|
# extract out members added to this group
|
|
$Members = $_.Properties.members | Select-Object -ExpandProperty Member | Where-Object { $_.action -match 'ADD' } | ForEach-Object {
|
|
if($_.sid) { $_.sid }
|
|
else { $_.name }
|
|
}
|
|
|
|
if ($Members) {
|
|
|
|
# extract out any/all filters...I hate you GPP
|
|
if($_.filters) {
|
|
$Filters = $_.filters.GetEnumerator() | ForEach-Object {
|
|
New-Object -TypeName PSObject -Property @{'Type' = $_.LocalName;'Value' = $_.name}
|
|
}
|
|
}
|
|
else {
|
|
$Filters = $Null
|
|
}
|
|
|
|
if($Members -isnot [System.Array]) { $Members = @($Members) }
|
|
|
|
$GPOGroup = New-Object PSObject
|
|
$GPOGroup | Add-Member Noteproperty 'GPOPath' $TargetGroupsXMLPath
|
|
$GPOGroup | Add-Member Noteproperty 'Filters' $Filters
|
|
$GPOGroup | Add-Member Noteproperty 'GroupName' $GroupName
|
|
$GPOGroup | Add-Member Noteproperty 'GroupSID' $GroupSID
|
|
$GPOGroup | Add-Member Noteproperty 'GroupMemberOf' $Null
|
|
$GPOGroup | Add-Member Noteproperty 'GroupMembers' $Members
|
|
$GPOGroup
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose "Error parsing $TargetGroupsXMLPath : $_"
|
|
}
|
|
|
|
if($UsePSDrive -and $RandDrive) {
|
|
Write-Verbose "Removing temp PSDrive $RandDrive"
|
|
Get-PSDrive -Name $RandDrive -ErrorAction SilentlyContinue | Remove-PSDrive -Force
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingGPO {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$GPOname = '*',
|
|
|
|
[String]
|
|
$DisplayName,
|
|
|
|
[String]
|
|
$ComputerName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
$GPOSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $ADSpath -PageSize $PageSize
|
|
}
|
|
|
|
process {
|
|
if ($GPOSearcher) {
|
|
|
|
if($ComputerName) {
|
|
$GPONames = @()
|
|
$Computers = Get-ThisThingComputer -ComputerName $ComputerName -Domain $Domain -DomainController $DomainController -FullData -PageSize $PageSize
|
|
|
|
if(!$Computers) {
|
|
throw "Computer $ComputerName in domain '$Domain' not found! Try a fully qualified host name"
|
|
}
|
|
|
|
# get the given computer's OU
|
|
$ComputerOUs = @()
|
|
ForEach($Computer in $Computers) {
|
|
# extract all OUs a computer is a part of
|
|
$DN = $Computer.distinguishedname
|
|
|
|
$ComputerOUs += $DN.split(",") | ForEach-Object {
|
|
if($_.startswith("OU=")) {
|
|
$DN.substring($DN.indexof($_))
|
|
}
|
|
}
|
|
}
|
|
|
|
Write-Verbose "ComputerOUs: $ComputerOUs"
|
|
|
|
# find all the GPOs linked to the computer's OU
|
|
ForEach($ComputerOU in $ComputerOUs) {
|
|
$GPONames += Get-ThisThingOU -Domain $Domain -DomainController $DomainController -ADSpath $ComputerOU -FullData -PageSize $PageSize | ForEach-Object {
|
|
# get any GPO links
|
|
write-verbose "blah: $($_.name)"
|
|
$_.gplink.split("][") | ForEach-Object {
|
|
if ($_.startswith("LDAP")) {
|
|
$_.split(";")[0]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Write-Verbose "GPONames: $GPONames"
|
|
|
|
# find any GPOs linked to the site for the given computer
|
|
$ComputerSite = (Get-SiteName -ComputerName $ComputerName).SiteName
|
|
if($ComputerSite -and ($ComputerSite -notlike 'Error*')) {
|
|
$GPONames += Get-ThisThingSite -SiteName $ComputerSite -FullData | ForEach-Object {
|
|
if($_.gplink) {
|
|
$_.gplink.split("][") | ForEach-Object {
|
|
if ($_.startswith("LDAP")) {
|
|
$_.split(";")[0]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$GPONames | Where-Object{$_ -and ($_ -ne '')} | ForEach-Object {
|
|
|
|
# use the gplink as an ADS path to enumerate all GPOs for the computer
|
|
$GPOSearcher = Get-DomainSearcher -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $_ -PageSize $PageSize
|
|
$GPOSearcher.filter="(&(objectCategory=groupPolicyContainer)(name=$GPOname))"
|
|
|
|
try {
|
|
$Results = $GPOSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
$Out = Convert-LDAPProperty -Properties $_.Properties
|
|
$Out | Add-Member Noteproperty 'ComputerName' $ComputerName
|
|
$Out
|
|
}
|
|
$Results.dispose()
|
|
$GPOSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Warning $_
|
|
}
|
|
}
|
|
}
|
|
|
|
else {
|
|
if($DisplayName) {
|
|
$GPOSearcher.filter="(&(objectCategory=groupPolicyContainer)(displayname=$DisplayName))"
|
|
}
|
|
else {
|
|
$GPOSearcher.filter="(&(objectCategory=groupPolicyContainer)(name=$GPOname))"
|
|
}
|
|
|
|
try {
|
|
$Results = $GPOSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
if($ADSPath -and ($ADSpath -Match '^GC://')) {
|
|
$Properties = Convert-LDAPProperty -Properties $_.Properties
|
|
try {
|
|
$GPODN = $Properties.distinguishedname
|
|
$GPODomain = $GPODN.subString($GPODN.IndexOf("DC=")) -replace 'DC=','' -replace ',','.'
|
|
$gpcfilesyspath = "\\$GPODomain\SysVol\$GPODomain\Policies\$($Properties.cn)"
|
|
$Properties | Add-Member Noteproperty 'gpcfilesyspath' $gpcfilesyspath
|
|
$Properties
|
|
}
|
|
catch {
|
|
$Properties
|
|
}
|
|
}
|
|
else {
|
|
# convert/process the LDAP fields for each result
|
|
Convert-LDAPProperty -Properties $_.Properties
|
|
}
|
|
}
|
|
$Results.dispose()
|
|
$GPOSearcher.dispose()
|
|
}
|
|
catch {
|
|
Write-Warning $_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function New-GPOImmediateTask {
|
|
[CmdletBinding(DefaultParameterSetName = 'Create')]
|
|
Param (
|
|
[Parameter(ParameterSetName = 'Create', Mandatory = $True)]
|
|
[String]
|
|
[ValidateNotNullOrEmpty()]
|
|
$TaskName,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[String]
|
|
[ValidateNotNullOrEmpty()]
|
|
$Command = 'powershell',
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[String]
|
|
[ValidateNotNullOrEmpty()]
|
|
$CommandArguments,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[String]
|
|
[ValidateNotNullOrEmpty()]
|
|
$TaskDescription = '',
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[String]
|
|
[ValidateNotNullOrEmpty()]
|
|
$TaskAuthor = 'NT AUTHORITY\System',
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[String]
|
|
[ValidateNotNullOrEmpty()]
|
|
$TaskModifiedDate = (Get-Date (Get-Date).AddDays(-30) -Format u).trim("Z"),
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[String]
|
|
$GPOname,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[String]
|
|
$GPODisplayName,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[String]
|
|
$Domain,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[Switch]
|
|
$Force,
|
|
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[Switch]
|
|
$Remove,
|
|
|
|
[Parameter(ParameterSetName = 'Create')]
|
|
[Parameter(ParameterSetName = 'Remove')]
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
# build the XML spec for our 'immediate' scheduled task
|
|
$TaskXML = '<?xml version="1.0" encoding="utf-8"?><ScheduledTasks clsid="{CC63F200-7309-4ba0-B154-A71CD118DBCC}"><ImmediateTaskV2 clsid="{9756B581-76EC-4169-9AFC-0CA8D43ADB5F}" name="'+$TaskName+'" image="0" changed="'+$TaskModifiedDate+'" uid="{'+$([guid]::NewGuid())+'}" userContext="0" removePolicy="0"><Properties action="C" name="'+$TaskName+'" runAs="NT AUTHORITY\System" logonType="S4U"><Task version="1.3"><RegistrationInfo><Author>'+$TaskAuthor+'</Author><Description>'+$TaskDescription+'</Description></RegistrationInfo><Principals><Principal id="Author"><UserId>NT AUTHORITY\System</UserId><RunLevel>HighestAvailable</RunLevel><LogonType>S4U</LogonType></Principal></Principals><Settings><IdleSettings><Duration>PT10M</Duration><WaitTimeout>PT1H</WaitTimeout><StopOnIdleEnd>true</StopOnIdleEnd><RestartOnIdle>false</RestartOnIdle></IdleSettings><MultipleInstancesPolicy>IgnoreNew</MultipleInstancesPolicy><DisallowStartIfOnBatteries>false</DisallowStartIfOnBatteries><StopIfGoingOnBatteries>true</StopIfGoingOnBatteries><AllowHardTerminate>false</AllowHardTerminate><StartWhenAvailable>true</StartWhenAvailable><AllowStartOnDemand>false</AllowStartOnDemand><Enabled>true</Enabled><Hidden>true</Hidden><ExecutionTimeLimit>PT0S</ExecutionTimeLimit><Priority>7</Priority><DeleteExpiredTaskAfter>PT0S</DeleteExpiredTaskAfter><RestartOnFailure><Interval>PT15M</Interval><Count>3</Count></RestartOnFailure></Settings><Actions Context="Author"><Exec><Command>'+$Command+'</Command><Arguments>'+$CommandArguments+'</Arguments></Exec></Actions><Triggers><TimeTrigger><StartBoundary>%LocalTimeXmlEx%</StartBoundary><EndBoundary>%LocalTimeXmlEx%</EndBoundary><Enabled>true</Enabled></TimeTrigger></Triggers></Task></Properties></ImmediateTaskV2></ScheduledTasks>'
|
|
|
|
if (!$PSBoundParameters['GPOname'] -and !$PSBoundParameters['GPODisplayName']) {
|
|
Write-Warning 'Either -GPOName or -GPODisplayName must be specified'
|
|
return
|
|
}
|
|
|
|
# eunmerate the specified GPO(s)
|
|
$GPOs = Get-ThisThingGPO -GPOname $GPOname -DisplayName $GPODisplayName -Domain $Domain -DomainController $DomainController -ADSpath $ADSpath -Credential $Credential
|
|
|
|
if(!$GPOs) {
|
|
Write-Warning 'No GPO found.'
|
|
return
|
|
}
|
|
|
|
$GPOs | ForEach-Object {
|
|
$ProcessedGPOName = $_.Name
|
|
try {
|
|
Write-Verbose "Trying to weaponize GPO: $ProcessedGPOName"
|
|
|
|
# map a network drive as New-PSDrive/New-Item/etc. don't accept -Credential properly :(
|
|
if($Credential) {
|
|
Write-Verbose "Mapping '$($_.gpcfilesyspath)' to network drive N:\"
|
|
$Path = $_.gpcfilesyspath.TrimEnd('\')
|
|
$Net = New-Object -ComObject WScript.Network
|
|
$Net.MapNetworkDrive("N:", $Path, $False, $Credential.UserName, $Credential.GetNetworkCredential().Password)
|
|
$TaskPath = "N:\Machine\Preferences\ScheduledTasks\"
|
|
}
|
|
else {
|
|
$TaskPath = $_.gpcfilesyspath + "\Machine\Preferences\ScheduledTasks\"
|
|
}
|
|
|
|
if($Remove) {
|
|
if(!(Test-Path "$TaskPath\ScheduledTasks.xml")) {
|
|
Throw "Scheduled task doesn't exist at $TaskPath\ScheduledTasks.xml"
|
|
}
|
|
|
|
if (!$Force -and !$psCmdlet.ShouldContinue('Do you want to continue?',"Removing schtask at $TaskPath\ScheduledTasks.xml")) {
|
|
return
|
|
}
|
|
|
|
Remove-Item -Path "$TaskPath\ScheduledTasks.xml" -Force
|
|
}
|
|
else {
|
|
if (!$Force -and !$psCmdlet.ShouldContinue('Do you want to continue?',"Creating schtask at $TaskPath\ScheduledTasks.xml")) {
|
|
return
|
|
}
|
|
|
|
# create the folder if it doesn't exist
|
|
$Null = New-Item -ItemType Directory -Force -Path $TaskPath
|
|
|
|
if(Test-Path "$TaskPath\ScheduledTasks.xml") {
|
|
Throw "Scheduled task already exists at $TaskPath\ScheduledTasks.xml !"
|
|
}
|
|
|
|
$TaskXML | Set-Content -Encoding ASCII -Path "$TaskPath\ScheduledTasks.xml"
|
|
}
|
|
|
|
if($Credential) {
|
|
Write-Verbose "Removing mounted drive at N:\"
|
|
$Net = New-Object -ComObject WScript.Network
|
|
$Net.RemoveNetworkDrive("N:")
|
|
}
|
|
}
|
|
catch {
|
|
Write-Warning "Error for GPO $ProcessedGPOName : $_"
|
|
if($Credential) {
|
|
Write-Verbose "Removing mounted drive at N:\"
|
|
$Net = New-Object -ComObject WScript.Network
|
|
$Net.RemoveNetworkDrive("N:")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingGPOGroup {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[String]
|
|
$GPOname = '*',
|
|
|
|
[String]
|
|
$DisplayName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[Switch]
|
|
$ResolveMemberSIDs,
|
|
|
|
[Switch]
|
|
$UsePSDrive,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
$Option = [System.StringSplitOptions]::RemoveEmptyEntries
|
|
|
|
# get every GPO from the specified domain with restricted groups set
|
|
Get-ThisThingGPO -GPOName $GPOname -DisplayName $DisplayName -Domain $Domain -DomainController $DomainController -ADSpath $ADSpath -PageSize $PageSize | ForEach-Object {
|
|
|
|
$GPOdisplayName = $_.displayname
|
|
$GPOname = $_.name
|
|
$GPOPath = $_.gpcfilesyspath
|
|
|
|
$ParseArgs = @{
|
|
'GptTmplPath' = "$GPOPath\MACHINE\Microsoft\Windows NT\SecEdit\GptTmpl.inf"
|
|
'UsePSDrive' = $UsePSDrive
|
|
}
|
|
|
|
# parse the GptTmpl.inf 'Restricted Groups' file if it exists
|
|
$Inf = Get-GptTmpl @ParseArgs
|
|
|
|
if($Inf -and ($Inf.psbase.Keys -contains 'Group Membership')) {
|
|
|
|
$Memberships = @{}
|
|
|
|
# group the members/memberof fields for each entry
|
|
ForEach ($Membership in $Inf.'Group Membership'.GetEnumerator()) {
|
|
$Group, $Relation = $Membership.Key.Split('__', $Option) | ForEach-Object {$_.Trim()}
|
|
|
|
# extract out ALL members
|
|
$MembershipValue = $Membership.Value | Where-Object {$_} | ForEach-Object { $_.Trim('*') } | Where-Object {$_}
|
|
|
|
if($ResolveMemberSIDs) {
|
|
# if the resulting member is username and not a SID, attempt to resolve it
|
|
$GroupMembers = @()
|
|
ForEach($Member in $MembershipValue) {
|
|
if($Member -and ($Member.Trim() -ne '')) {
|
|
if($Member -notmatch '^S-1-.*') {
|
|
$MemberSID = Convert-NameToSid -Domain $Domain -ObjectName $Member | Select-Object -ExpandProperty SID
|
|
if($MemberSID) {
|
|
$GroupMembers += $MemberSID
|
|
}
|
|
else {
|
|
$GroupMembers += $Member
|
|
}
|
|
}
|
|
else {
|
|
$GroupMembers += $Member
|
|
}
|
|
}
|
|
}
|
|
$MembershipValue = $GroupMembers
|
|
}
|
|
|
|
if(-not $Memberships[$Group]) {
|
|
$Memberships[$Group] = @{}
|
|
}
|
|
if($MembershipValue -isnot [System.Array]) {$MembershipValue = @($MembershipValue)}
|
|
$Memberships[$Group].Add($Relation, $MembershipValue)
|
|
}
|
|
|
|
ForEach ($Membership in $Memberships.GetEnumerator()) {
|
|
if($Membership -and $Membership.Key -and ($Membership.Key -match '^\*')) {
|
|
# if the SID is already resolved (i.e. begins with *) try to resolve SID to a name
|
|
$GroupSID = $Membership.Key.Trim('*')
|
|
if($GroupSID -and ($GroupSID.Trim() -ne '')) {
|
|
$GroupName = Convert-SidToName -SID $GroupSID
|
|
}
|
|
else {
|
|
$GroupName = $False
|
|
}
|
|
}
|
|
else {
|
|
$GroupName = $Membership.Key
|
|
|
|
if($GroupName -and ($GroupName.Trim() -ne '')) {
|
|
if($Groupname -match 'Administrators') {
|
|
$GroupSID = 'S-1-5-32-544'
|
|
}
|
|
elseif($Groupname -match 'Remote Desktop') {
|
|
$GroupSID = 'S-1-5-32-555'
|
|
}
|
|
elseif($Groupname -match 'Guests') {
|
|
$GroupSID = 'S-1-5-32-546'
|
|
}
|
|
elseif($GroupName.Trim() -ne '') {
|
|
$GroupSID = Convert-NameToSid -Domain $Domain -ObjectName $Groupname | Select-Object -ExpandProperty SID
|
|
}
|
|
else {
|
|
$GroupSID = $Null
|
|
}
|
|
}
|
|
}
|
|
|
|
$GPOGroup = New-Object PSObject
|
|
$GPOGroup | Add-Member Noteproperty 'GPODisplayName' $GPODisplayName
|
|
$GPOGroup | Add-Member Noteproperty 'GPOName' $GPOName
|
|
$GPOGroup | Add-Member Noteproperty 'GPOPath' $GPOPath
|
|
$GPOGroup | Add-Member Noteproperty 'GPOType' 'RestrictedGroups'
|
|
$GPOGroup | Add-Member Noteproperty 'Filters' $Null
|
|
$GPOGroup | Add-Member Noteproperty 'GroupName' $GroupName
|
|
$GPOGroup | Add-Member Noteproperty 'GroupSID' $GroupSID
|
|
$GPOGroup | Add-Member Noteproperty 'GroupMemberOf' $Membership.Value.Memberof
|
|
$GPOGroup | Add-Member Noteproperty 'GroupMembers' $Membership.Value.Members
|
|
$GPOGroup
|
|
}
|
|
}
|
|
|
|
$ParseArgs = @{
|
|
'GroupsXMLpath' = "$GPOPath\MACHINE\Preferences\Groups\Groups.xml"
|
|
'UsePSDrive' = $UsePSDrive
|
|
}
|
|
|
|
Get-GroupsXML @ParseArgs | ForEach-Object {
|
|
if($ResolveMemberSIDs) {
|
|
$GroupMembers = @()
|
|
ForEach($Member in $_.GroupMembers) {
|
|
if($Member -and ($Member.Trim() -ne '')) {
|
|
if($Member -notmatch '^S-1-.*') {
|
|
# if the resulting member is username and not a SID, attempt to resolve it
|
|
$MemberSID = Convert-NameToSid -Domain $Domain -ObjectName $Member | Select-Object -ExpandProperty SID
|
|
if($MemberSID) {
|
|
$GroupMembers += $MemberSID
|
|
}
|
|
else {
|
|
$GroupMembers += $Member
|
|
}
|
|
}
|
|
else {
|
|
$GroupMembers += $Member
|
|
}
|
|
}
|
|
}
|
|
$_.GroupMembers = $GroupMembers
|
|
}
|
|
|
|
$_ | Add-Member Noteproperty 'GPODisplayName' $GPODisplayName
|
|
$_ | Add-Member Noteproperty 'GPOName' $GPOName
|
|
$_ | Add-Member Noteproperty 'GPOType' 'GroupPolicyPreferences'
|
|
$_
|
|
}
|
|
}
|
|
}
|
|
|
|
function Find-GPOLocation {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[String]
|
|
$UserName,
|
|
|
|
[String]
|
|
$GroupName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$LocalGroup = 'Administrators',
|
|
|
|
[Switch]
|
|
$UsePSDrive,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
if($UserName) {
|
|
# if a group name is specified, get that user object so we can extract the target SID
|
|
$User = Get-ThisThingUser -UserName $UserName -Domain $Domain -DomainController $DomainController -PageSize $PageSize | Select-Object -First 1
|
|
$UserSid = $User.objectsid
|
|
|
|
if(-not $UserSid) {
|
|
Throw "User '$UserName' not found!"
|
|
}
|
|
|
|
$TargetSIDs = @($UserSid)
|
|
$ObjectSamAccountName = $User.samaccountname
|
|
$TargetObject = $UserSid
|
|
}
|
|
elseif($GroupName) {
|
|
# if a group name is specified, get that group object so we can extract the target SID
|
|
$Group = Get-ThisThingGroup -GroupName $GroupName -Domain $Domain -DomainController $DomainController -FullData -PageSize $PageSize | Select-Object -First 1
|
|
$GroupSid = $Group.objectsid
|
|
|
|
if(-not $GroupSid) {
|
|
Throw "Group '$GroupName' not found!"
|
|
}
|
|
|
|
$TargetSIDs = @($GroupSid)
|
|
$ObjectSamAccountName = $Group.samaccountname
|
|
$TargetObject = $GroupSid
|
|
}
|
|
else {
|
|
$TargetSIDs = @('*')
|
|
}
|
|
|
|
# figure out what the SID is of the target local group we're checking for membership in
|
|
if($LocalGroup -like "*Admin*") {
|
|
$TargetLocalSID = 'S-1-5-32-544'
|
|
}
|
|
elseif ( ($LocalGroup -like "*RDP*") -or ($LocalGroup -like "*Remote*") ) {
|
|
$TargetLocalSID = 'S-1-5-32-555'
|
|
}
|
|
elseif ($LocalGroup -like "S-1-5-*") {
|
|
$TargetLocalSID = $LocalGroup
|
|
}
|
|
else {
|
|
throw "LocalGroup must be 'Administrators', 'RDP', or a 'S-1-5-X' SID format."
|
|
}
|
|
|
|
# if we're not listing all relationships, use the tokenGroups approach from Get-ThisThingGroup to
|
|
# get all effective security SIDs this object is a part of
|
|
if($TargetSIDs[0] -and ($TargetSIDs[0] -ne '*')) {
|
|
$TargetSIDs += Get-ThisThingGroup -Domain $Domain -DomainController $DomainController -PageSize $PageSize -UserName $ObjectSamAccountName -RawSids
|
|
}
|
|
|
|
if(-not $TargetSIDs) {
|
|
throw "No effective target SIDs!"
|
|
}
|
|
|
|
Write-Verbose "TargetLocalSID: $TargetLocalSID"
|
|
Write-Verbose "Effective target SIDs: $TargetSIDs"
|
|
|
|
$GPOGroupArgs = @{
|
|
'Domain' = $Domain
|
|
'DomainController' = $DomainController
|
|
'UsePSDrive' = $UsePSDrive
|
|
'ResolveMemberSIDs' = $True
|
|
'PageSize' = $PageSize
|
|
}
|
|
|
|
# enumerate all GPO group mappings for the target domain that involve our target SID set
|
|
$GPOgroups = Get-ThisThingGPOGroup @GPOGroupArgs | ForEach-Object {
|
|
|
|
$GPOgroup = $_
|
|
|
|
# if the locally set group is what we're looking for, check the GroupMembers ('members')
|
|
# for our target SID
|
|
if($GPOgroup.GroupSID -match $TargetLocalSID) {
|
|
$GPOgroup.GroupMembers | Where-Object {$_} | ForEach-Object {
|
|
if ( ($TargetSIDs[0] -eq '*') -or ($TargetSIDs -Contains $_) ) {
|
|
$GPOgroup
|
|
}
|
|
}
|
|
}
|
|
# if the group is a 'memberof' the group we're looking for, check GroupSID against the targt SIDs
|
|
if( ($GPOgroup.GroupMemberOf -contains $TargetLocalSID) ) {
|
|
if( ($TargetSIDs[0] -eq '*') -or ($TargetSIDs -Contains $GPOgroup.GroupSID) ) {
|
|
$GPOgroup
|
|
}
|
|
}
|
|
} | Sort-Object -Property GPOName -Unique
|
|
|
|
$GPOgroups | ForEach-Object {
|
|
|
|
$GPOname = $_.GPODisplayName
|
|
$GPOguid = $_.GPOName
|
|
$GPOPath = $_.GPOPath
|
|
$GPOType = $_.GPOType
|
|
if($_.GroupMembers) {
|
|
$GPOMembers = $_.GroupMembers
|
|
}
|
|
else {
|
|
$GPOMembers = $_.GroupSID
|
|
}
|
|
|
|
$Filters = $_.Filters
|
|
|
|
if(-not $TargetObject) {
|
|
# if the * wildcard was used, set the ObjectDistName as the GPO member SID set
|
|
# so all relationship mappings are output
|
|
$TargetObjectSIDs = $GPOMembers
|
|
}
|
|
else {
|
|
$TargetObjectSIDs = $TargetObject
|
|
}
|
|
|
|
# find any OUs that have this GUID applied and then retrieve any computers from the OU
|
|
Get-ThisThingOU -Domain $Domain -DomainController $DomainController -GUID $GPOguid -FullData -PageSize $PageSize | ForEach-Object {
|
|
if($Filters) {
|
|
# filter for computer name/org unit if a filter is specified
|
|
# TODO: handle other filters (i.e. OU filters?) again, I hate you GPP...
|
|
$OUComputers = Get-ThisThingComputer -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $_.ADSpath -FullData -PageSize $PageSize | Where-Object {
|
|
$_.adspath -match ($Filters.Value)
|
|
} | ForEach-Object { $_.dnshostname }
|
|
}
|
|
else {
|
|
$OUComputers = Get-ThisThingComputer -Domain $Domain -DomainController $DomainController -Credential $Credential -ADSpath $_.ADSpath -PageSize $PageSize
|
|
}
|
|
|
|
if($OUComputers) {
|
|
if($OUComputers -isnot [System.Array]) {$OUComputers = @($OUComputers)}
|
|
|
|
ForEach ($TargetSid in $TargetObjectSIDs) {
|
|
$Object = Get-ADObject -SID $TargetSid -Domain $Domain -DomainController $DomainController -Credential $Credential -PageSize $PageSize
|
|
|
|
$IsGroup = @('268435456','268435457','536870912','536870913') -contains $Object.samaccounttype
|
|
|
|
$GPOLocation = New-Object PSObject
|
|
$GPOLocation | Add-Member Noteproperty 'ObjectName' $Object.samaccountname
|
|
$GPOLocation | Add-Member Noteproperty 'ObjectDN' $Object.distinguishedname
|
|
$GPOLocation | Add-Member Noteproperty 'ObjectSID' $Object.objectsid
|
|
$GPOLocation | Add-Member Noteproperty 'Domain' $Domain
|
|
$GPOLocation | Add-Member Noteproperty 'IsGroup' $IsGroup
|
|
$GPOLocation | Add-Member Noteproperty 'GPODisplayName' $GPOname
|
|
$GPOLocation | Add-Member Noteproperty 'GPOGuid' $GPOGuid
|
|
$GPOLocation | Add-Member Noteproperty 'GPOPath' $GPOPath
|
|
$GPOLocation | Add-Member Noteproperty 'GPOType' $GPOType
|
|
$GPOLocation | Add-Member Noteproperty 'ContainerName' $_.distinguishedname
|
|
$GPOLocation | Add-Member Noteproperty 'ComputerName' $OUComputers
|
|
$GPOLocation.PSObject.TypeNames.Add('PowerView.GPOLocalGroup')
|
|
$GPOLocation
|
|
}
|
|
}
|
|
}
|
|
|
|
# find any sites that have this GUID applied
|
|
Get-ThisThingSite -Domain $Domain -DomainController $DomainController -GUID $GPOguid -PageSize $PageSize -FullData | ForEach-Object {
|
|
|
|
ForEach ($TargetSid in $TargetObjectSIDs) {
|
|
$Object = Get-ADObject -SID $TargetSid -Domain $Domain -DomainController $DomainController -Credential $Credential -PageSize $PageSize
|
|
|
|
$IsGroup = @('268435456','268435457','536870912','536870913') -contains $Object.samaccounttype
|
|
|
|
$AppliedSite = New-Object PSObject
|
|
$AppliedSite | Add-Member Noteproperty 'ObjectName' $Object.samaccountname
|
|
$AppliedSite | Add-Member Noteproperty 'ObjectDN' $Object.distinguishedname
|
|
$AppliedSite | Add-Member Noteproperty 'ObjectSID' $Object.objectsid
|
|
$AppliedSite | Add-Member Noteproperty 'IsGroup' $IsGroup
|
|
$AppliedSite | Add-Member Noteproperty 'Domain' $Domain
|
|
$AppliedSite | Add-Member Noteproperty 'GPODisplayName' $GPOname
|
|
$AppliedSite | Add-Member Noteproperty 'GPOGuid' $GPOGuid
|
|
$AppliedSite | Add-Member Noteproperty 'GPOPath' $GPOPath
|
|
$AppliedSite | Add-Member Noteproperty 'GPOType' $GPOType
|
|
$AppliedSite | Add-Member Noteproperty 'ContainerName' $_.distinguishedname
|
|
$AppliedSite | Add-Member Noteproperty 'ComputerName' $_.siteobjectbl
|
|
$AppliedSite.PSObject.TypeNames.Add('PowerView.GPOLocalGroup')
|
|
$AppliedSite
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Find-GPOComputerAdmin {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$ComputerName,
|
|
|
|
[String]
|
|
$OUName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$Recurse,
|
|
|
|
[String]
|
|
$LocalGroup = 'Administrators',
|
|
|
|
[Switch]
|
|
$UsePSDrive,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
process {
|
|
|
|
if(!$ComputerName -and !$OUName) {
|
|
Throw "-ComputerName or -OUName must be provided"
|
|
}
|
|
|
|
$GPOGroups = @()
|
|
|
|
if($ComputerName) {
|
|
$Computers = Get-ThisThingComputer -ComputerName $ComputerName -Domain $Domain -DomainController $DomainController -FullData -PageSize $PageSize
|
|
|
|
if(!$Computers) {
|
|
throw "Computer $ComputerName in domain '$Domain' not found! Try a fully qualified host name"
|
|
}
|
|
|
|
$TargetOUs = @()
|
|
ForEach($Computer in $Computers) {
|
|
# extract all OUs a computer is a part of
|
|
$DN = $Computer.distinguishedname
|
|
|
|
$TargetOUs += $DN.split(",") | ForEach-Object {
|
|
if($_.startswith("OU=")) {
|
|
$DN.substring($DN.indexof($_))
|
|
}
|
|
}
|
|
}
|
|
|
|
# enumerate any linked GPOs for the computer's site
|
|
$ComputerSite = (Get-SiteName -ComputerName $ComputerName).SiteName
|
|
if($ComputerSite -and ($ComputerSite -notlike 'Error*')) {
|
|
$GPOGroups += Get-ThisThingSite -SiteName $ComputerSite -FullData | ForEach-Object {
|
|
if($_.gplink) {
|
|
$_.gplink.split("][") | ForEach-Object {
|
|
if ($_.startswith("LDAP")) {
|
|
$_.split(";")[0]
|
|
}
|
|
}
|
|
}
|
|
} | ForEach-Object {
|
|
$GPOGroupArgs = @{
|
|
'Domain' = $Domain
|
|
'DomainController' = $DomainController
|
|
'ResolveMemberSIDs' = $True
|
|
'UsePSDrive' = $UsePSDrive
|
|
'PageSize' = $PageSize
|
|
}
|
|
|
|
# for each GPO link, get any locally set user/group SIDs
|
|
Get-ThisThingGPOGroup @GPOGroupArgs
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
$TargetOUs = @($OUName)
|
|
}
|
|
|
|
Write-Verbose "Target OUs: $TargetOUs"
|
|
|
|
$TargetOUs | Where-Object {$_} | ForEach-Object {
|
|
|
|
$GPOLinks = Get-ThisThingOU -Domain $Domain -DomainController $DomainController -ADSpath $_ -FullData -PageSize $PageSize | ForEach-Object {
|
|
# and then get any GPO links
|
|
if($_.gplink) {
|
|
$_.gplink.split("][") | ForEach-Object {
|
|
if ($_.startswith("LDAP")) {
|
|
$_.split(";")[0]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$GPOGroupArgs = @{
|
|
'Domain' = $Domain
|
|
'DomainController' = $DomainController
|
|
'UsePSDrive' = $UsePSDrive
|
|
'ResolveMemberSIDs' = $True
|
|
'PageSize' = $PageSize
|
|
}
|
|
|
|
# extract GPO groups that are set through any gPlink for this OU
|
|
$GPOGroups += Get-ThisThingGPOGroup @GPOGroupArgs | ForEach-Object {
|
|
ForEach($GPOLink in $GPOLinks) {
|
|
$Name = $_.GPOName
|
|
if($GPOLink -like "*$Name*") {
|
|
$_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# for each found GPO group, resolve the SIDs of the members
|
|
$GPOgroups | Sort-Object -Property GPOName -Unique | ForEach-Object {
|
|
$GPOGroup = $_
|
|
|
|
if($GPOGroup.GroupMembers) {
|
|
$GPOMembers = $GPOGroup.GroupMembers
|
|
}
|
|
else {
|
|
$GPOMembers = $GPOGroup.GroupSID
|
|
}
|
|
|
|
$GPOMembers | ForEach-Object {
|
|
# resolve this SID to a domain object
|
|
$Object = Get-ADObject -Domain $Domain -DomainController $DomainController -PageSize $PageSize -SID $_
|
|
|
|
$IsGroup = @('268435456','268435457','536870912','536870913') -contains $Object.samaccounttype
|
|
|
|
$GPOComputerAdmin = New-Object PSObject
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ComputerName' $ComputerName
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ObjectName' $Object.samaccountname
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ObjectDN' $Object.distinguishedname
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ObjectSID' $_
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'IsGroup' $IsGroup
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPODisplayName' $GPOGroup.GPODisplayName
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPOGuid' $GPOGroup.GPOName
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPOPath' $GPOGroup.GPOPath
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPOType' $GPOGroup.GPOType
|
|
$GPOComputerAdmin
|
|
|
|
# if we're recursing and the current result object is a group
|
|
if($Recurse -and $GPOComputerAdmin.isGroup) {
|
|
|
|
Get-ThisThingGroupMember -Domain $Domain -DomainController $DomainController -SID $_ -FullData -Recurse -PageSize $PageSize | ForEach-Object {
|
|
|
|
$MemberDN = $_.distinguishedName
|
|
|
|
# extract the FQDN from the Distinguished Name
|
|
$MemberDomain = $MemberDN.subString($MemberDN.IndexOf("DC=")) -replace 'DC=','' -replace ',','.'
|
|
|
|
$MemberIsGroup = @('268435456','268435457','536870912','536870913') -contains $_.samaccounttype
|
|
|
|
if ($_.samAccountName) {
|
|
# forest users have the samAccountName set
|
|
$MemberName = $_.samAccountName
|
|
}
|
|
else {
|
|
# external trust users have a SID, so convert it
|
|
try {
|
|
$MemberName = Convert-SidToName $_.cn
|
|
}
|
|
catch {
|
|
# if there's a problem contacting the domain to resolve the SID
|
|
$MemberName = $_.cn
|
|
}
|
|
}
|
|
|
|
$GPOComputerAdmin = New-Object PSObject
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ComputerName' $ComputerName
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ObjectName' $MemberName
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ObjectDN' $MemberDN
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'ObjectSID' $_.objectsid
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'IsGroup' $MemberIsGrou
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPODisplayName' $GPOGroup.GPODisplayName
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPOGuid' $GPOGroup.GPOName
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPOPath' $GPOGroup.GPOPath
|
|
$GPOComputerAdmin | Add-Member Noteproperty 'GPOType' $GPOTypep
|
|
$GPOComputerAdmin
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-DomainPolicy {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[String]
|
|
[ValidateSet("Domain","DC")]
|
|
$Source ="Domain",
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$ResolveSids,
|
|
|
|
[Switch]
|
|
$UsePSDrive
|
|
)
|
|
|
|
if($Source -eq "Domain") {
|
|
# query the given domain for the default domain policy object
|
|
$GPO = Get-ThisThingGPO -Domain $Domain -DomainController $DomainController -GPOname "{31B2F340-016D-11D2-945F-00C04FB984F9}"
|
|
|
|
if($GPO) {
|
|
# grab the GptTmpl.inf file and parse it
|
|
$GptTmplPath = $GPO.gpcfilesyspath + "\MACHINE\Microsoft\Windows NT\SecEdit\GptTmpl.inf"
|
|
|
|
$ParseArgs = @{
|
|
'GptTmplPath' = $GptTmplPath
|
|
'UsePSDrive' = $UsePSDrive
|
|
}
|
|
|
|
# parse the GptTmpl.inf
|
|
Get-GptTmpl @ParseArgs
|
|
}
|
|
|
|
}
|
|
elseif($Source -eq "DC") {
|
|
# query the given domain/dc for the default domain controller policy object
|
|
$GPO = Get-ThisThingGPO -Domain $Domain -DomainController $DomainController -GPOname "{6AC1786C-016F-11D2-945F-00C04FB984F9}"
|
|
|
|
if($GPO) {
|
|
# grab the GptTmpl.inf file and parse it
|
|
$GptTmplPath = $GPO.gpcfilesyspath + "\MACHINE\Microsoft\Windows NT\SecEdit\GptTmpl.inf"
|
|
|
|
$ParseArgs = @{
|
|
'GptTmplPath' = $GptTmplPath
|
|
'UsePSDrive' = $UsePSDrive
|
|
}
|
|
|
|
# parse the GptTmpl.inf
|
|
Get-GptTmpl @ParseArgs | ForEach-Object {
|
|
if($ResolveSids) {
|
|
# if we're resolving sids in PrivilegeRights to names
|
|
$Policy = New-Object PSObject
|
|
$_.psobject.properties | ForEach-Object {
|
|
if( $_.Name -eq 'PrivilegeRights') {
|
|
|
|
$PrivilegeRights = New-Object PSObject
|
|
# for every nested SID member of PrivilegeRights, try to unpack everything and resolve the SIDs as appropriate
|
|
$_.Value.psobject.properties | ForEach-Object {
|
|
|
|
$Sids = $_.Value | ForEach-Object {
|
|
try {
|
|
if($_ -isnot [System.Array]) {
|
|
Convert-SidToName $_
|
|
}
|
|
else {
|
|
$_ | ForEach-Object { Convert-SidToName $_ }
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose "Error resolving SID : $_"
|
|
}
|
|
}
|
|
|
|
$PrivilegeRights | Add-Member Noteproperty $_.Name $Sids
|
|
}
|
|
|
|
$Policy | Add-Member Noteproperty 'PrivilegeRights' $PrivilegeRights
|
|
}
|
|
else {
|
|
$Policy | Add-Member Noteproperty $_.Name $_.Value
|
|
}
|
|
}
|
|
$Policy
|
|
}
|
|
else { $_ }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function Get-ThisThingLocalGroup {
|
|
[CmdletBinding(DefaultParameterSetName = 'WinNT')]
|
|
param(
|
|
[Parameter(ParameterSetName = 'API', Position=0, ValueFromPipeline=$True)]
|
|
[Parameter(ParameterSetName = 'WinNT', Position=0, ValueFromPipeline=$True)]
|
|
[Alias('HostName')]
|
|
[String[]]
|
|
$ComputerName = $Env:ComputerName,
|
|
|
|
[Parameter(ParameterSetName = 'WinNT')]
|
|
[Parameter(ParameterSetName = 'API')]
|
|
[ValidateScript({Test-Path -Path $_ })]
|
|
[Alias('HostList')]
|
|
[String]
|
|
$ComputerFile,
|
|
|
|
[Parameter(ParameterSetName = 'WinNT')]
|
|
[Parameter(ParameterSetName = 'API')]
|
|
[String]
|
|
$GroupName = 'Administrators',
|
|
|
|
[Parameter(ParameterSetName = 'WinNT')]
|
|
[Switch]
|
|
$ListGroups,
|
|
|
|
[Parameter(ParameterSetName = 'WinNT')]
|
|
[Switch]
|
|
$Recurse,
|
|
|
|
[Parameter(ParameterSetName = 'API')]
|
|
[Switch]
|
|
$API
|
|
)
|
|
|
|
process {
|
|
|
|
$Servers = @()
|
|
|
|
# if we have a host list passed, grab it
|
|
if($ComputerFile) {
|
|
$Servers = Get-Content -Path $ComputerFile
|
|
}
|
|
else {
|
|
# otherwise assume a single host name
|
|
$Servers += $ComputerName | Get-NameField
|
|
}
|
|
|
|
# query the specified group using the WINNT provider, and
|
|
# extract fields as appropriate from the results
|
|
ForEach($Server in $Servers) {
|
|
|
|
if($API) {
|
|
# if we're using the Netapi32 NetLocalGroupGetMembers API call to get the local group information
|
|
# arguments for NetLocalGroupGetMembers
|
|
$QueryLevel = 2
|
|
$PtrInfo = [IntPtr]::Zero
|
|
$EntriesRead = 0
|
|
$TotalRead = 0
|
|
$ResumeHandle = 0
|
|
|
|
# get the local user information
|
|
$Result = $Netapi32::NetLocalGroupGetMembers($Server, $GroupName, $QueryLevel, [ref]$PtrInfo, -1, [ref]$EntriesRead, [ref]$TotalRead, [ref]$ResumeHandle)
|
|
|
|
# Locate the offset of the initial intPtr
|
|
$Offset = $PtrInfo.ToInt64()
|
|
|
|
$LocalUsers = @()
|
|
|
|
# 0 = success
|
|
if (($Result -eq 0) -and ($Offset -gt 0)) {
|
|
|
|
# Work out how much to increment the pointer by finding out the size of the structure
|
|
$Increment = $LOCALGROUP_MEMBERS_INFO_2::GetSize()
|
|
|
|
# parse all the result structures
|
|
for ($i = 0; ($i -lt $EntriesRead); $i++) {
|
|
# create a new int ptr at the given offset and cast the pointer as our result structure
|
|
$NewIntPtr = New-Object System.Intptr -ArgumentList $Offset
|
|
$Info = $NewIntPtr -as $LOCALGROUP_MEMBERS_INFO_2
|
|
|
|
$Offset = $NewIntPtr.ToInt64()
|
|
$Offset += $Increment
|
|
|
|
$SidString = ""
|
|
$Result2 = $Advapi32::ConvertSidToStringSid($Info.lgrmi2_sid, [ref]$SidString);$LastError = [Runtime.InteropServices.Marshal]::GetLastWin32Error()
|
|
|
|
if($Result2 -eq 0) {
|
|
Write-Verbose "Error: $(([ComponentModel.Win32Exception] $LastError).Message)"
|
|
}
|
|
else {
|
|
$LocalUser = New-Object PSObject
|
|
$LocalUser | Add-Member Noteproperty 'ComputerName' $Server
|
|
$LocalUser | Add-Member Noteproperty 'AccountName' $Info.lgrmi2_domainandname
|
|
$LocalUser | Add-Member Noteproperty 'SID' $SidString
|
|
|
|
$IsGroup = $($Info.lgrmi2_sidusage -eq 'SidTypeGroup')
|
|
$LocalUser | Add-Member Noteproperty 'IsGroup' $IsGroup
|
|
$LocalUser.PSObject.TypeNames.Add('PowerView.LocalUserAPI')
|
|
|
|
$LocalUsers += $LocalUser
|
|
}
|
|
}
|
|
|
|
# free up the result buffer
|
|
$Null = $Netapi32::NetApiBufferFree($PtrInfo)
|
|
|
|
# try to extract out the machine SID by using the -500 account as a reference
|
|
$MachineSid = $LocalUsers | Where-Object {$_.SID -like '*-500'}
|
|
$Parts = $MachineSid.SID.Split('-')
|
|
$MachineSid = $Parts[0..($Parts.Length -2)] -join '-'
|
|
|
|
$LocalUsers | ForEach-Object {
|
|
if($_.SID -match $MachineSid) {
|
|
$_ | Add-Member Noteproperty 'IsDomain' $False
|
|
}
|
|
else {
|
|
$_ | Add-Member Noteproperty 'IsDomain' $True
|
|
}
|
|
}
|
|
$LocalUsers
|
|
}
|
|
else {
|
|
Write-Verbose "Error: $(([ComponentModel.Win32Exception] $Result).Message)"
|
|
}
|
|
}
|
|
|
|
else {
|
|
# otherwise we're using the WinNT service provider
|
|
try {
|
|
if($ListGroups) {
|
|
# if we're listing the group names on a remote server
|
|
$Computer = [ADSI]"WinNT://$Server,computer"
|
|
|
|
$Computer.psbase.children | Where-Object { $_.psbase.schemaClassName -eq 'group' } | ForEach-Object {
|
|
$Group = New-Object PSObject
|
|
$Group | Add-Member Noteproperty 'Server' $Server
|
|
$Group | Add-Member Noteproperty 'Group' ($_.name[0])
|
|
$Group | Add-Member Noteproperty 'SID' ((New-Object System.Security.Principal.SecurityIdentifier $_.objectsid[0],0).Value)
|
|
$Group | Add-Member Noteproperty 'Description' ($_.Description[0])
|
|
$Group.PSObject.TypeNames.Add('PowerView.LocalGroup')
|
|
$Group
|
|
}
|
|
}
|
|
else {
|
|
# otherwise we're listing the group members
|
|
$Members = @($([ADSI]"WinNT://$Server/$GroupName,group").psbase.Invoke('Members'))
|
|
|
|
$Members | ForEach-Object {
|
|
|
|
$Member = New-Object PSObject
|
|
$Member | Add-Member Noteproperty 'ComputerName' $Server
|
|
|
|
$AdsPath = ($_.GetType().InvokeMember('Adspath', 'GetProperty', $Null, $_, $Null)).Replace('WinNT://', '')
|
|
$Class = $_.GetType().InvokeMember('Class', 'GetProperty', $Null, $_, $Null)
|
|
|
|
# try to translate the NT4 domain to a FQDN if possible
|
|
$Name = Convert-ADName -ObjectName $AdsPath -InputType 'NT4' -OutputType 'Canonical'
|
|
$IsGroup = $Class -eq "Group"
|
|
|
|
if($Name) {
|
|
$FQDN = $Name.split("/")[0]
|
|
$ObjName = $AdsPath.split("/")[-1]
|
|
$Name = "$FQDN/$ObjName"
|
|
$IsDomain = $True
|
|
}
|
|
else {
|
|
$ObjName = $AdsPath.split("/")[-1]
|
|
$Name = $AdsPath
|
|
$IsDomain = $False
|
|
}
|
|
|
|
$Member | Add-Member Noteproperty 'AccountName' $Name
|
|
$Member | Add-Member Noteproperty 'IsDomain' $IsDomain
|
|
$Member | Add-Member Noteproperty 'IsGroup' $IsGroup
|
|
|
|
if($IsDomain) {
|
|
# translate the binary sid to a string
|
|
$Member | Add-Member Noteproperty 'SID' ((New-Object System.Security.Principal.SecurityIdentifier($_.GetType().InvokeMember('ObjectSID', 'GetProperty', $Null, $_, $Null),0)).Value)
|
|
$Member | Add-Member Noteproperty 'Description' ""
|
|
$Member | Add-Member Noteproperty 'Disabled' ""
|
|
|
|
if($IsGroup) {
|
|
$Member | Add-Member Noteproperty 'LastLogin' ""
|
|
}
|
|
else {
|
|
try {
|
|
$Member | Add-Member Noteproperty 'LastLogin' ( $_.GetType().InvokeMember('LastLogin', 'GetProperty', $Null, $_, $Null))
|
|
}
|
|
catch {
|
|
$Member | Add-Member Noteproperty 'LastLogin' ""
|
|
}
|
|
}
|
|
$Member | Add-Member Noteproperty 'PwdLastSet' ""
|
|
$Member | Add-Member Noteproperty 'PwdExpired' ""
|
|
$Member | Add-Member Noteproperty 'UserFlags' ""
|
|
}
|
|
else {
|
|
# repull this user object so we can ensure correct information
|
|
$LocalUser = $([ADSI] "WinNT://$AdsPath")
|
|
|
|
# translate the binary sid to a string
|
|
$Member | Add-Member Noteproperty 'SID' ((New-Object System.Security.Principal.SecurityIdentifier($LocalUser.objectSid.value,0)).Value)
|
|
$Member | Add-Member Noteproperty 'Description' ($LocalUser.Description[0])
|
|
|
|
if($IsGroup) {
|
|
$Member | Add-Member Noteproperty 'PwdLastSet' ""
|
|
$Member | Add-Member Noteproperty 'PwdExpired' ""
|
|
$Member | Add-Member Noteproperty 'UserFlags' ""
|
|
$Member | Add-Member Noteproperty 'Disabled' ""
|
|
$Member | Add-Member Noteproperty 'LastLogin' ""
|
|
}
|
|
else {
|
|
$Member | Add-Member Noteproperty 'PwdLastSet' ( (Get-Date).AddSeconds(-$LocalUser.PasswordAge[0]))
|
|
$Member | Add-Member Noteproperty 'PwdExpired' ( $LocalUser.PasswordExpired[0] -eq '1')
|
|
$Member | Add-Member Noteproperty 'UserFlags' ( $LocalUser.UserFlags[0] )
|
|
# UAC flags of 0x2 mean the account is disabled
|
|
$Member | Add-Member Noteproperty 'Disabled' $(($LocalUser.userFlags.value -band 2) -eq 2)
|
|
try {
|
|
$Member | Add-Member Noteproperty 'LastLogin' ( $LocalUser.LastLogin[0])
|
|
}
|
|
catch {
|
|
$Member | Add-Member Noteproperty 'LastLogin' ""
|
|
}
|
|
}
|
|
}
|
|
$Member.PSObject.TypeNames.Add('PowerView.LocalUser')
|
|
$Member
|
|
|
|
# if the result is a group domain object and we're recursing,
|
|
# try to resolve all the group member results
|
|
if($Recurse -and $IsGroup) {
|
|
if($IsDomain) {
|
|
$FQDN = $Name.split("/")[0]
|
|
$GroupName = $Name.split("/")[1].trim()
|
|
|
|
Get-ThisThingGroupMember -GroupName $GroupName -Domain $FQDN -FullData -Recurse | ForEach-Object {
|
|
|
|
$Member = New-Object PSObject
|
|
$Member | Add-Member Noteproperty 'ComputerName' "$FQDN/$($_.GroupName)"
|
|
|
|
$MemberDN = $_.distinguishedName
|
|
# extract the FQDN from the Distinguished Name
|
|
$MemberDomain = $MemberDN.subString($MemberDN.IndexOf("DC=")) -replace 'DC=','' -replace ',','.'
|
|
|
|
$MemberIsGroup = @('268435456','268435457','536870912','536870913') -contains $_.samaccounttype
|
|
|
|
if ($_.samAccountName) {
|
|
# forest users have the samAccountName set
|
|
$MemberName = $_.samAccountName
|
|
}
|
|
else {
|
|
try {
|
|
# external trust users have a SID, so convert it
|
|
try {
|
|
$MemberName = Convert-SidToName $_.cn
|
|
}
|
|
catch {
|
|
# if there's a problem contacting the domain to resolve the SID
|
|
$MemberName = $_.cn
|
|
}
|
|
}
|
|
catch {
|
|
Write-Debug "Error resolving SID : $_"
|
|
}
|
|
}
|
|
|
|
$Member | Add-Member Noteproperty 'AccountName' "$MemberDomain/$MemberName"
|
|
$Member | Add-Member Noteproperty 'SID' $_.objectsid
|
|
$Member | Add-Member Noteproperty 'Description' $_.description
|
|
$Member | Add-Member Noteproperty 'Disabled' $False
|
|
$Member | Add-Member Noteproperty 'IsGroup' $MemberIsGroup
|
|
$Member | Add-Member Noteproperty 'IsDomain' $True
|
|
$Member | Add-Member Noteproperty 'LastLogin' ''
|
|
$Member | Add-Member Noteproperty 'PwdLastSet' $_.pwdLastSet
|
|
$Member | Add-Member Noteproperty 'PwdExpired' ''
|
|
$Member | Add-Member Noteproperty 'UserFlags' $_.userAccountControl
|
|
$Member.PSObject.TypeNames.Add('PowerView.LocalUser')
|
|
$Member
|
|
}
|
|
} else {
|
|
Get-ThisThingLocalGroup -ComputerName $Server -GroupName $ObjName -Recurse
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Warning "[!] Error: $_"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
filter Get-MySMBShare {
|
|
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[Alias('HostName')]
|
|
[Object[]]
|
|
[ValidateNotNullOrEmpty()]
|
|
$ComputerName = 'localhost'
|
|
)
|
|
|
|
# extract the computer name from whatever object was passed on the pipeline
|
|
$Computer = $ComputerName | Get-NameField
|
|
|
|
# arguments for NetShareEnum
|
|
$QueryLevel = 1
|
|
$PtrInfo = [IntPtr]::Zero
|
|
$EntriesRead = 0
|
|
$TotalRead = 0
|
|
$ResumeHandle = 0
|
|
|
|
# get the share information
|
|
$Result = $Netapi32::NetShareEnum($Computer, $QueryLevel, [ref]$PtrInfo, -1, [ref]$EntriesRead, [ref]$TotalRead, [ref]$ResumeHandle)
|
|
|
|
# Locate the offset of the initial intPtr
|
|
$Offset = $PtrInfo.ToInt64()
|
|
|
|
# 0 = success
|
|
if (($Result -eq 0) -and ($Offset -gt 0)) {
|
|
|
|
# Work out how much to increment the pointer by finding out the size of the structure
|
|
$Increment = $SHARE_INFO_1::GetSize()
|
|
|
|
# parse all the result structures
|
|
for ($i = 0; ($i -lt $EntriesRead); $i++) {
|
|
# create a new int ptr at the given offset and cast the pointer as our result structure
|
|
$NewIntPtr = New-Object System.Intptr -ArgumentList $Offset
|
|
$Info = $NewIntPtr -as $SHARE_INFO_1
|
|
|
|
# return all the sections of the structure
|
|
$Shares = $Info | Select-Object *
|
|
$Shares | Add-Member Noteproperty 'ComputerName' $Computer
|
|
$Offset = $NewIntPtr.ToInt64()
|
|
$Offset += $Increment
|
|
|
|
# Get ip address of host
|
|
$targethostname = $Shares.ComputerName
|
|
$ComputerIpAddress = [System.Net.Dns]::GetHostAddresses("$targethostname") | select IPAddressToString -first 1 -ExpandProperty IPAddressToString
|
|
|
|
$ShareObject = New-Object -TypeName PSObject
|
|
$ShareObject | Add-Member NoteProperty "ComputerName" $Shares.ComputerName
|
|
$ShareObject | Add-Member NoteProperty "IpAddress" $ComputerIpAddress
|
|
$ShareObject | Add-Member NoteProperty "ShareName" $Shares.shi1_netname
|
|
$ShareObject | Add-Member NoteProperty "ShareDesc" $Shares.shi1_remark
|
|
$ShareObject | Add-Member NoteProperty "Sharetype" $Shares.shi1_type
|
|
|
|
$ComputerName = $Shares.ComputerName
|
|
$ShareName = $Shares.shi1_netname
|
|
$ShareType = $Shares.shi1_type
|
|
$ShareDesc = $Shares.shi1_remark
|
|
|
|
# Check access
|
|
try{
|
|
$TargetPath = "\\$ComputerName\$ShareName"
|
|
$Null = [IO.Directory]::GetFiles($TargetPath)
|
|
Write-Verbose "$Computer : ACCESSIBLE! - Share: \\$computerName\$ShareName Desc: $ShareDesc Type:$ShareType"
|
|
$ShareObject | Add-Member NoteProperty "ShareAccess" "Yes"
|
|
$ShareObject
|
|
}catch{
|
|
Write-Verbose "$Computer : NOT ACCESSIBLE - Share: \\$computerName\$ShareName Desc: $ShareDesc Type:$ShareType"
|
|
$ShareObject | Add-Member NoteProperty "ShareAccess" "No"
|
|
$ShareObject
|
|
}
|
|
}
|
|
|
|
# free up the result buffer
|
|
$Null = $Netapi32::NetApiBufferFree($PtrInfo)
|
|
}
|
|
else {
|
|
Write-Verbose "Error: $(([ComponentModel.Win32Exception] $Result).Message)"
|
|
}
|
|
}
|
|
|
|
function Find-InterestingFile {
|
|
<#
|
|
.SYNOPSIS
|
|
|
|
This function recursively searches a given UNC path for files with
|
|
specific keywords in the name (default of pass, sensitive, secret, admin,
|
|
login and unattend*.xml). The output can be piped out to a csv with the
|
|
-OutFile flag. By default, hidden files/folders are included in search results.
|
|
|
|
.PARAMETER Path
|
|
|
|
UNC/local path to recursively search.
|
|
|
|
.PARAMETER Terms
|
|
|
|
Terms to search for.
|
|
|
|
.PARAMETER OfficeDocs
|
|
|
|
Switch. Search for office documents (*.doc*, *.xls*, *.ppt*)
|
|
|
|
.PARAMETER FreshEXEs
|
|
|
|
Switch. Find .EXEs accessed within the last week.
|
|
|
|
.PARAMETER LastAccessTime
|
|
|
|
Only return files with a LastAccessTime greater than this date value.
|
|
|
|
.PARAMETER LastWriteTime
|
|
|
|
Only return files with a LastWriteTime greater than this date value.
|
|
|
|
.PARAMETER CreationTime
|
|
|
|
Only return files with a CreationTime greater than this date value.
|
|
|
|
.PARAMETER ExcludeFolders
|
|
|
|
Switch. Exclude folders from the search results.
|
|
|
|
.PARAMETER ExcludeHidden
|
|
|
|
Switch. Exclude hidden files and folders from the search results.
|
|
|
|
.PARAMETER CheckWriteAccess
|
|
|
|
Switch. Only returns files the current user has write access to.
|
|
|
|
.PARAMETER OutFile
|
|
|
|
Output results to a specified csv output file.
|
|
|
|
.PARAMETER UsePSDrive
|
|
|
|
Switch. Mount target remote path with temporary PSDrives.
|
|
|
|
.OUTPUTS
|
|
|
|
The full path, owner, lastaccess time, lastwrite time, and size for each found file.
|
|
|
|
.EXAMPLE
|
|
|
|
PS C:\> Find-InterestingFile -Path C:\Backup\
|
|
|
|
Returns any files on the local path C:\Backup\ that have the default
|
|
search term set in the title.
|
|
|
|
.EXAMPLE
|
|
|
|
PS C:\> Find-InterestingFile -Path \\WINDOWS7\Users\ -Terms salaries,email -OutFile out.csv
|
|
|
|
Returns any files on the remote path \\WINDOWS7\Users\ that have 'salaries'
|
|
or 'email' in the title, and writes the results out to a csv file
|
|
named 'out.csv'
|
|
|
|
.EXAMPLE
|
|
|
|
PS C:\> Find-InterestingFile -Path \\WINDOWS7\Users\ -LastAccessTime (Get-Date).AddDays(-7)
|
|
|
|
Returns any files on the remote path \\WINDOWS7\Users\ that have the default
|
|
search term set in the title and were accessed within the last week.
|
|
|
|
.LINK
|
|
|
|
http://www.harmj0y.net/blog/redteaming/file-server-triage-on-red-team-engagements/
|
|
#>
|
|
|
|
param(
|
|
[Parameter(ValueFromPipeline=$True)]
|
|
[String]
|
|
$Path = '.\',
|
|
|
|
[Alias('Terms')]
|
|
[String[]]
|
|
$SearchTerms = @('pass', 'sensitive', 'admin', 'login', 'secret', 'unattend*.xml', '.vmdk', 'creds', 'credential', '.config'),
|
|
|
|
[Switch]
|
|
$OfficeDocs,
|
|
|
|
[Switch]
|
|
$FreshEXEs,
|
|
|
|
[String]
|
|
$LastAccessTime,
|
|
|
|
[String]
|
|
$LastWriteTime,
|
|
|
|
[String]
|
|
$CreationTime,
|
|
|
|
[Switch]
|
|
$ExcludeFolders,
|
|
|
|
[Switch]
|
|
$ExcludeHidden,
|
|
|
|
[Switch]
|
|
$CheckWriteAccess,
|
|
|
|
[String]
|
|
$OutFile,
|
|
|
|
[Switch]
|
|
$UsePSDrive
|
|
)
|
|
|
|
begin {
|
|
|
|
$Path += if(!$Path.EndsWith('\')) {"\"}
|
|
|
|
if ($Credential) {
|
|
$UsePSDrive = $True
|
|
}
|
|
|
|
# append wildcards to the front and back of all search terms
|
|
$SearchTerms = $SearchTerms | ForEach-Object { if($_ -notmatch '^\*.*\*$') {"*$($_)*"} else{$_} }
|
|
|
|
# search just for office documents if specified
|
|
if ($OfficeDocs) {
|
|
$SearchTerms = @('*.doc', '*.docx', '*.xls', '*.xlsx', '*.ppt', '*.pptx')
|
|
}
|
|
|
|
# find .exe's accessed within the last 7 days
|
|
if($FreshEXEs) {
|
|
# get an access time limit of 7 days ago
|
|
$LastAccessTime = (Get-Date).AddDays(-7).ToString('MM/dd/yyyy')
|
|
$SearchTerms = '*.exe'
|
|
}
|
|
|
|
if($UsePSDrive) {
|
|
# if we're PSDrives, create a temporary mount point
|
|
|
|
$Parts = $Path.split('\')
|
|
$FolderPath = $Parts[0..($Parts.length-2)] -join '\'
|
|
$FilePath = $Parts[-1]
|
|
|
|
$RandDrive = ("abcdefghijklmnopqrstuvwxyz".ToCharArray() | Get-Random -Count 7) -join ''
|
|
|
|
Write-Verbose "Mounting path '$Path' using a temp PSDrive at $RandDrive"
|
|
|
|
try {
|
|
$Null = New-PSDrive -Name $RandDrive -PSProvider FileSystem -Root $FolderPath -ErrorAction Stop
|
|
}
|
|
catch {
|
|
Write-Verbose "Error mounting path '$Path' : $_"
|
|
return $Null
|
|
}
|
|
|
|
# so we can cd/dir the new drive
|
|
$Path = "${RandDrive}:\${FilePath}"
|
|
}
|
|
}
|
|
|
|
process {
|
|
|
|
Write-Verbose "[*] Search path $Path"
|
|
|
|
function Invoke-CheckWrite {
|
|
# short helper to check is the current user can write to a file
|
|
[CmdletBinding()]param([String]$Path)
|
|
try {
|
|
$Filetest = [IO.FILE]::OpenWrite($Path)
|
|
$Filetest.Close()
|
|
$True
|
|
}
|
|
catch {
|
|
Write-Verbose -Message $Error[0]
|
|
$False
|
|
}
|
|
}
|
|
|
|
$SearchArgs = @{
|
|
'Path' = $Path
|
|
'Recurse' = $True
|
|
'Force' = $(-not $ExcludeHidden)
|
|
'Include' = $SearchTerms
|
|
'ErrorAction' = 'SilentlyContinue'
|
|
}
|
|
|
|
Get-ChildItem @SearchArgs | ForEach-Object {
|
|
Write-Verbose $_
|
|
# check if we're excluding folders
|
|
if(!$ExcludeFolders -or !$_.PSIsContainer) {$_}
|
|
} | ForEach-Object {
|
|
if($LastAccessTime -or $LastWriteTime -or $CreationTime) {
|
|
if($LastAccessTime -and ($_.LastAccessTime -gt $LastAccessTime)) {$_}
|
|
elseif($LastWriteTime -and ($_.LastWriteTime -gt $LastWriteTime)) {$_}
|
|
elseif($CreationTime -and ($_.CreationTime -gt $CreationTime)) {$_}
|
|
}
|
|
else {$_}
|
|
} | ForEach-Object {
|
|
# filter for write access (if applicable)
|
|
if((-not $CheckWriteAccess) -or (Invoke-CheckWrite -Path $_.FullName)) {$_}
|
|
} | Select-Object FullName,@{Name='Owner';Expression={(Get-Acl $_.FullName).Owner}},LastAccessTime,LastWriteTime,CreationTime,Length | ForEach-Object {
|
|
# check if we're outputting to the pipeline or an output file
|
|
if($OutFile) {Export-PowerViewCSV -InputObject $_ -OutFile $OutFile}
|
|
else {$_}
|
|
}
|
|
}
|
|
|
|
end {
|
|
if($UsePSDrive -and $RandDrive) {
|
|
Write-Verbose "Removing temp PSDrive $RandDrive"
|
|
Get-PSDrive -Name $RandDrive -ErrorAction SilentlyContinue | Remove-PSDrive -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
function Invoke-ThreadedFunction {
|
|
# Helper used by any threaded host enumeration functions
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0,Mandatory=$True)]
|
|
[String[]]
|
|
$ComputerName,
|
|
|
|
[Parameter(Position=1,Mandatory=$True)]
|
|
[System.Management.Automation.ScriptBlock]
|
|
$ScriptBlock,
|
|
|
|
[Parameter(Position=2)]
|
|
[Hashtable]
|
|
$ScriptParameters,
|
|
|
|
[Int]
|
|
[ValidateRange(1,100)]
|
|
$Threads = 20,
|
|
|
|
[Switch]
|
|
$NoImports
|
|
)
|
|
|
|
begin {
|
|
|
|
if ($PSBoundParameters['Debug']) {
|
|
$DebugPreference = 'Continue'
|
|
}
|
|
|
|
Write-Verbose "[*] Total number of hosts: $($ComputerName.count)"
|
|
|
|
# Adapted from:
|
|
# http://powershell.org/wp/forums/topic/invpke-parallel-need-help-to-clone-the-current-runspace/
|
|
$SessionState = [System.Management.Automation.Runspaces.InitialSessionState]::CreateDefault()
|
|
$SessionState.ApartmentState = [System.Threading.Thread]::CurrentThread.GetApartmentState()
|
|
|
|
# import the current session state's variables and functions so the chained PowerView
|
|
# functionality can be used by the threaded blocks
|
|
if(!$NoImports) {
|
|
|
|
# grab all the current variables for this runspace
|
|
$MyVars = Get-Variable -Scope 2
|
|
|
|
# these Variables are added by Runspace.Open() Method and produce Stop errors if you add them twice
|
|
$VorbiddenVars = @("?","args","ConsoleFileName","Error","ExecutionContext","false","HOME","Host","input","InputObject","MaximumAliasCount","MaximumDriveCount","MaximumErrorCount","MaximumFunctionCount","MaximumHistoryCount","MaximumVariableCount","MyInvocation","null","PID","PSBoundParameters","PSCommandPath","PSCulture","PSDefaultParameterValues","PSHOME","PSScriptRoot","PSUICulture","PSVersionTable","PWD","ShellId","SynchronizedHash","true")
|
|
|
|
# Add Variables from Parent Scope (current runspace) into the InitialSessionState
|
|
ForEach($Var in $MyVars) {
|
|
if($VorbiddenVars -NotContains $Var.Name) {
|
|
$SessionState.Variables.Add((New-Object -TypeName System.Management.Automation.Runspaces.SessionStateVariableEntry -ArgumentList $Var.name,$Var.Value,$Var.description,$Var.options,$Var.attributes))
|
|
}
|
|
}
|
|
|
|
# Add Functions from current runspace to the InitialSessionState
|
|
ForEach($Function in (Get-ChildItem Function:)) {
|
|
$SessionState.Commands.Add((New-Object -TypeName System.Management.Automation.Runspaces.SessionStateFunctionEntry -ArgumentList $Function.Name, $Function.Definition))
|
|
}
|
|
}
|
|
|
|
# threading adapted from
|
|
# https://github.com/darkoperator/Posh-SecMod/blob/master/Discovery/Discovery.psm1#L407
|
|
# Thanks Carlos!
|
|
|
|
# create a pool of maxThread runspaces
|
|
$Pool = [runspacefactory]::CreateRunspacePool(1, $Threads, $SessionState, $Host)
|
|
$Pool.Open()
|
|
|
|
$method = $null
|
|
ForEach ($m in [PowerShell].GetMethods() | Where-Object { $_.Name -eq "BeginInvoke" }) {
|
|
$methodParameters = $m.GetParameters()
|
|
if (($methodParameters.Count -eq 2) -and $methodParameters[0].Name -eq "input" -and $methodParameters[1].Name -eq "output") {
|
|
$method = $m.MakeGenericMethod([Object], [Object])
|
|
break
|
|
}
|
|
}
|
|
|
|
$Jobs = @()
|
|
}
|
|
|
|
process {
|
|
|
|
ForEach ($Computer in $ComputerName) {
|
|
|
|
# make sure we get a server name
|
|
if ($Computer -ne '') {
|
|
# Write-Verbose "[*] Enumerating server $Computer ($($Counter+1) of $($ComputerName.count))"
|
|
|
|
While ($($Pool.GetAvailableRunspaces()) -le 0) {
|
|
Start-Sleep -MilliSeconds 500
|
|
}
|
|
|
|
# create a "powershell pipeline runner"
|
|
$p = [powershell]::create()
|
|
|
|
$p.runspacepool = $Pool
|
|
|
|
# add the script block + arguments
|
|
$Null = $p.AddScript($ScriptBlock).AddParameter('ComputerName', $Computer)
|
|
if($ScriptParameters) {
|
|
ForEach ($Param in $ScriptParameters.GetEnumerator()) {
|
|
$Null = $p.AddParameter($Param.Name, $Param.Value)
|
|
}
|
|
}
|
|
|
|
$o = New-Object Management.Automation.PSDataCollection[Object]
|
|
|
|
$Jobs += @{
|
|
PS = $p
|
|
Output = $o
|
|
Result = $method.Invoke($p, @($null, [Management.Automation.PSDataCollection[Object]]$o))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
end {
|
|
Write-Verbose "Waiting for threads to finish..."
|
|
|
|
Do {
|
|
ForEach ($Job in $Jobs) {
|
|
$Job.Output.ReadAll()
|
|
}
|
|
} While (($Jobs | Where-Object { ! $_.Result.IsCompleted }).Count -gt 0)
|
|
|
|
ForEach ($Job in $Jobs) {
|
|
$Job.PS.Dispose()
|
|
}
|
|
|
|
$Pool.Dispose()
|
|
Write-Verbose "All threads completed!"
|
|
}
|
|
}
|
|
|
|
|
|
|
|
function Invoke-ShareFinder {
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0,ValueFromPipeline=$True)]
|
|
[Alias('Hosts')]
|
|
[String[]]
|
|
$ComputerName,
|
|
|
|
[ValidateScript({Test-Path -Path $_ })]
|
|
[Alias('HostList')]
|
|
[String]
|
|
$ComputerFile,
|
|
|
|
[String]
|
|
$ComputerFilter,
|
|
|
|
[String]
|
|
$ComputerADSpath,
|
|
|
|
[Switch]
|
|
$ExcludeStandard,
|
|
|
|
[Switch]
|
|
$ExcludePrint,
|
|
|
|
[Switch]
|
|
$ExcludeIPC,
|
|
|
|
[Switch]
|
|
$NoPing,
|
|
|
|
[Switch]
|
|
$CheckShareAccess,
|
|
|
|
[Switch]
|
|
$CheckAdmin,
|
|
|
|
[UInt32]
|
|
$Delay = 0,
|
|
|
|
[Double]
|
|
$Jitter = .3,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$SearchForest,
|
|
|
|
[ValidateRange(1,100)]
|
|
[Int]
|
|
$Threads
|
|
)
|
|
|
|
begin {
|
|
if ($PSBoundParameters['Debug']) {
|
|
$DebugPreference = 'Continue'
|
|
}
|
|
|
|
# random object for delay
|
|
$RandNo = New-Object System.Random
|
|
|
|
Write-Verbose "[*] Running Invoke-ShareFinder with delay of $Delay"
|
|
|
|
# figure out the shares we want to ignore
|
|
[String[]] $ExcludedShares = @('')
|
|
|
|
if ($ExcludePrint) {
|
|
$ExcludedShares = $ExcludedShares + "PRINT$"
|
|
}
|
|
if ($ExcludeIPC) {
|
|
$ExcludedShares = $ExcludedShares + "IPC$"
|
|
}
|
|
if ($ExcludeStandard) {
|
|
$ExcludedShares = @('', "ADMIN$", "IPC$", "C$", "PRINT$")
|
|
}
|
|
|
|
# if we're using a host file list, read the targets in and add them to the target list
|
|
if($ComputerFile) {
|
|
$ComputerName = Get-Content -Path $ComputerFile
|
|
}
|
|
|
|
if(!$ComputerName) {
|
|
[array]$ComputerName = @()
|
|
|
|
if($Domain) {
|
|
$TargetDomains = @($Domain)
|
|
}
|
|
elseif($SearchForest) {
|
|
# get ALL the domains in the forest to search
|
|
$TargetDomains = Get-ThisThingForestDomain | ForEach-Object { $_.Name }
|
|
}
|
|
else {
|
|
# use the local domain
|
|
$TargetDomains = @( (Get-ThisThingDomain).name )
|
|
}
|
|
|
|
ForEach ($Domain in $TargetDomains) {
|
|
Write-Verbose "[*] Querying domain $Domain for hosts"
|
|
$ComputerName += Get-ThisThingComputer -Domain $Domain -DomainController $DomainController -Filter $ComputerFilter -ADSpath $ComputerADSpath
|
|
}
|
|
|
|
# remove any null target hosts, uniquify the list and shuffle it
|
|
$ComputerName = $ComputerName | Where-Object { $_ } | Sort-Object -Unique | Sort-Object { Get-Random }
|
|
if($($ComputerName.count) -eq 0) {
|
|
throw "No hosts found!"
|
|
}
|
|
}
|
|
|
|
# script block that enumerates a server
|
|
$HostEnumBlock = {
|
|
param($ComputerName, $Ping, $CheckShareAccess, $ExcludedShares, $CheckAdmin)
|
|
|
|
# optionally check if the server is up first
|
|
$Up = $True
|
|
if($Ping) {
|
|
$Up = Test-Connection -Count 1 -Quiet -ComputerName $ComputerName
|
|
}
|
|
if($Up) {
|
|
# get the shares for this host and check what we find
|
|
$Shares = Get-MySMBShare -ComputerName $ComputerName
|
|
ForEach ($Share in $Shares) {
|
|
Write-Verbose "[*] Server share: $Share"
|
|
$NetName = $Share.shi1_netname
|
|
$Remark = $Share.shi1_remark
|
|
$Path = '\\'+$ComputerName+'\'+$NetName
|
|
|
|
# make sure we get a real share name back
|
|
if (($NetName) -and ($NetName.trim() -ne '')) {
|
|
# if we're just checking for access to ADMIN$
|
|
if($CheckAdmin) {
|
|
if($NetName.ToUpper() -eq "ADMIN$") {
|
|
try {
|
|
$Null = [IO.Directory]::GetFiles($Path)
|
|
"\\$ComputerName\$NetName `t- $Remark"
|
|
}
|
|
catch {
|
|
Write-Verbose "Error accessing path $Path : $_"
|
|
}
|
|
}
|
|
}
|
|
# skip this share if it's in the exclude list
|
|
elseif ($ExcludedShares -NotContains $NetName.ToUpper()) {
|
|
# see if we want to check access to this share
|
|
if($CheckShareAccess) {
|
|
# check if the user has access to this path
|
|
try {
|
|
$Null = [IO.Directory]::GetFiles($Path)
|
|
"\\$ComputerName\$NetName `t- $Remark"
|
|
}
|
|
catch {
|
|
Write-Verbose "Error accessing path $Path : $_"
|
|
}
|
|
}
|
|
else {
|
|
"\\$ComputerName\$NetName `t- $Remark"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
process {
|
|
|
|
if($Threads) {
|
|
Write-Verbose "Using threading with threads = $Threads"
|
|
|
|
# if we're using threading, kick off the script block with Invoke-ThreadedFunction
|
|
$ScriptParams = @{
|
|
'Ping' = $(-not $NoPing)
|
|
'CheckShareAccess' = $CheckShareAccess
|
|
'ExcludedShares' = $ExcludedShares
|
|
'CheckAdmin' = $CheckAdmin
|
|
}
|
|
|
|
# kick off the threaded script block + arguments
|
|
Invoke-ThreadedFunction -ComputerName $ComputerName -ScriptBlock $HostEnumBlock -ScriptParameters $ScriptParams -Threads $Threads
|
|
}
|
|
|
|
else {
|
|
if(-not $NoPing -and ($ComputerName.count -ne 1)) {
|
|
# ping all hosts in parallel
|
|
$Ping = {param($ComputerName) if(Test-Connection -ComputerName $ComputerName -Count 1 -Quiet -ErrorAction Stop){$ComputerName}}
|
|
$ComputerName = Invoke-ThreadedFunction -NoImports -ComputerName $ComputerName -ScriptBlock $Ping -Threads 100
|
|
}
|
|
|
|
Write-Verbose "[*] Total number of active hosts: $($ComputerName.count)"
|
|
$Counter = 0
|
|
|
|
ForEach ($Computer in $ComputerName) {
|
|
|
|
$Counter = $Counter + 1
|
|
|
|
# sleep for our semi-randomized interval
|
|
Start-Sleep -Seconds $RandNo.Next((1-$Jitter)*$Delay, (1+$Jitter)*$Delay)
|
|
|
|
Write-Verbose "[*] Enumerating server $Computer ($Counter of $($ComputerName.count))"
|
|
Invoke-Command -ScriptBlock $HostEnumBlock -ArgumentList $Computer, $False, $CheckShareAccess, $ExcludedShares, $CheckAdmin
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
|
|
function Invoke-FileFinder {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0,ValueFromPipeline=$True)]
|
|
[Alias('Hosts')]
|
|
[String[]]
|
|
$ComputerName,
|
|
|
|
[ValidateScript({Test-Path -Path $_ })]
|
|
[Alias('HostList')]
|
|
[String]
|
|
$ComputerFile,
|
|
|
|
[String]
|
|
$ComputerFilter,
|
|
|
|
[String]
|
|
$ComputerADSpath,
|
|
|
|
[ValidateScript({Test-Path -Path $_ })]
|
|
[String]
|
|
$ShareList,
|
|
|
|
[Switch]
|
|
$OfficeDocs,
|
|
|
|
[Switch]
|
|
$FreshEXEs,
|
|
|
|
[Alias('Terms')]
|
|
[String[]]
|
|
$SearchTerms,
|
|
|
|
[ValidateScript({Test-Path -Path $_ })]
|
|
[String]
|
|
$TermList,
|
|
|
|
[String]
|
|
$LastAccessTime,
|
|
|
|
[String]
|
|
$LastWriteTime,
|
|
|
|
[String]
|
|
$CreationTime,
|
|
|
|
[Switch]
|
|
$IncludeC,
|
|
|
|
[Switch]
|
|
$IncludeAdmin,
|
|
|
|
[Switch]
|
|
$ExcludeFolders,
|
|
|
|
[Switch]
|
|
$ExcludeHidden,
|
|
|
|
[Switch]
|
|
$CheckWriteAccess,
|
|
|
|
[String]
|
|
$OutFile,
|
|
|
|
[Switch]
|
|
$NoClobber,
|
|
|
|
[Switch]
|
|
$NoPing,
|
|
|
|
[UInt32]
|
|
$Delay = 0,
|
|
|
|
[Double]
|
|
$Jitter = .3,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$SearchForest,
|
|
|
|
[Switch]
|
|
$SearchSYSVOL,
|
|
|
|
[ValidateRange(1,100)]
|
|
[Int]
|
|
$Threads,
|
|
|
|
[Switch]
|
|
$UsePSDrive
|
|
)
|
|
|
|
begin {
|
|
if ($PSBoundParameters['Debug']) {
|
|
$DebugPreference = 'Continue'
|
|
}
|
|
|
|
# random object for delay
|
|
$RandNo = New-Object System.Random
|
|
|
|
Write-Verbose "[*] Running Invoke-FileFinder with delay of $Delay"
|
|
|
|
$Shares = @()
|
|
|
|
# figure out the shares we want to ignore
|
|
[String[]] $ExcludedShares = @("C$", "ADMIN$")
|
|
|
|
# see if we're specifically including any of the normally excluded sets
|
|
if ($IncludeC) {
|
|
if ($IncludeAdmin) {
|
|
$ExcludedShares = @()
|
|
}
|
|
else {
|
|
$ExcludedShares = @("ADMIN$")
|
|
}
|
|
}
|
|
|
|
if ($IncludeAdmin) {
|
|
if ($IncludeC) {
|
|
$ExcludedShares = @()
|
|
}
|
|
else {
|
|
$ExcludedShares = @("C$")
|
|
}
|
|
}
|
|
|
|
# delete any existing output file if it already exists
|
|
if(!$NoClobber) {
|
|
if ($OutFile -and (Test-Path -Path $OutFile)) { Remove-Item -Path $OutFile }
|
|
}
|
|
|
|
# if there's a set of terms specified to search for
|
|
if ($TermList) {
|
|
ForEach ($Term in Get-Content -Path $TermList) {
|
|
if (($Term -ne $Null) -and ($Term.trim() -ne '')) {
|
|
$SearchTerms += $Term
|
|
}
|
|
}
|
|
}
|
|
|
|
# if we're hard-passed a set of shares
|
|
if($ShareList) {
|
|
ForEach ($Item in Get-Content -Path $ShareList) {
|
|
if (($Item -ne $Null) -and ($Item.trim() -ne '')) {
|
|
# exclude any "[tab]- commants", i.e. the output from Invoke-ShareFinder
|
|
$Share = $Item.Split("`t")[0]
|
|
$Shares += $Share
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
# if we're using a host file list, read the targets in and add them to the target list
|
|
if($ComputerFile) {
|
|
$ComputerName = Get-Content -Path $ComputerFile
|
|
}
|
|
|
|
if(!$ComputerName) {
|
|
|
|
if($Domain) {
|
|
$TargetDomains = @($Domain)
|
|
}
|
|
elseif($SearchForest) {
|
|
# get ALL the domains in the forest to search
|
|
$TargetDomains = Get-ThisThingForestDomain | ForEach-Object { $_.Name }
|
|
}
|
|
else {
|
|
# use the local domain
|
|
$TargetDomains = @( (Get-ThisThingDomain).name )
|
|
}
|
|
|
|
if($SearchSYSVOL) {
|
|
ForEach ($Domain in $TargetDomains) {
|
|
$DCSearchPath = "\\$Domain\SYSVOL\"
|
|
Write-Verbose "[*] Adding share search path $DCSearchPath"
|
|
$Shares += $DCSearchPath
|
|
}
|
|
if(!$SearchTerms) {
|
|
# search for interesting scripts on SYSVOL
|
|
$SearchTerms = @('.vbs', '.bat', '.ps1')
|
|
}
|
|
}
|
|
else {
|
|
[array]$ComputerName = @()
|
|
|
|
ForEach ($Domain in $TargetDomains) {
|
|
Write-Verbose "[*] Querying domain $Domain for hosts"
|
|
$ComputerName += Get-ThisThingComputer -Filter $ComputerFilter -ADSpath $ComputerADSpath -Domain $Domain -DomainController $DomainController
|
|
}
|
|
|
|
# remove any null target hosts, uniquify the list and shuffle it
|
|
$ComputerName = $ComputerName | Where-Object { $_ } | Sort-Object -Unique | Sort-Object { Get-Random }
|
|
if($($ComputerName.Count) -eq 0) {
|
|
throw "No hosts found!"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# script block that enumerates shares and files on a server
|
|
$HostEnumBlock = {
|
|
param($ComputerName, $Ping, $ExcludedShares, $SearchTerms, $ExcludeFolders, $OfficeDocs, $ExcludeHidden, $FreshEXEs, $CheckWriteAccess, $OutFile, $UsePSDrive)
|
|
|
|
Write-Verbose "ComputerName: $ComputerName"
|
|
Write-Verbose "ExcludedShares: $ExcludedShares"
|
|
$SearchShares = @()
|
|
|
|
if($ComputerName.StartsWith("\\")) {
|
|
# if a share is passed as the server
|
|
$SearchShares += $ComputerName
|
|
}
|
|
else {
|
|
# if we're enumerating the shares on the target server first
|
|
$Up = $True
|
|
if($Ping) {
|
|
$Up = Test-Connection -Count 1 -Quiet -ComputerName $ComputerName
|
|
}
|
|
if($Up) {
|
|
# get the shares for this host and display what we find
|
|
$Shares = Get-MySMBShare -ComputerName $ComputerName
|
|
ForEach ($Share in $Shares) {
|
|
|
|
$NetName = $Share.shi1_netname
|
|
$Path = '\\'+$ComputerName+'\'+$NetName
|
|
|
|
# make sure we get a real share name back
|
|
if (($NetName) -and ($NetName.trim() -ne '')) {
|
|
|
|
# skip this share if it's in the exclude list
|
|
if ($ExcludedShares -NotContains $NetName.ToUpper()) {
|
|
# check if the user has access to this path
|
|
try {
|
|
$Null = [IO.Directory]::GetFiles($Path)
|
|
$SearchShares += $Path
|
|
}
|
|
catch {
|
|
Write-Verbose "[!] No access to $Path"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
ForEach($Share in $SearchShares) {
|
|
$SearchArgs = @{
|
|
'Path' = $Share
|
|
'SearchTerms' = $SearchTerms
|
|
'OfficeDocs' = $OfficeDocs
|
|
'FreshEXEs' = $FreshEXEs
|
|
'LastAccessTime' = $LastAccessTime
|
|
'LastWriteTime' = $LastWriteTime
|
|
'CreationTime' = $CreationTime
|
|
'ExcludeFolders' = $ExcludeFolders
|
|
'ExcludeHidden' = $ExcludeHidden
|
|
'CheckWriteAccess' = $CheckWriteAccess
|
|
'OutFile' = $OutFile
|
|
'UsePSDrive' = $UsePSDrive
|
|
}
|
|
|
|
Find-InterestingFile @SearchArgs
|
|
}
|
|
}
|
|
}
|
|
|
|
process {
|
|
|
|
if($Threads) {
|
|
Write-Verbose "Using threading with threads = $Threads"
|
|
|
|
# if we're using threading, kick off the script block with Invoke-ThreadedFunction
|
|
$ScriptParams = @{
|
|
'Ping' = $(-not $NoPing)
|
|
'ExcludedShares' = $ExcludedShares
|
|
'SearchTerms' = $SearchTerms
|
|
'ExcludeFolders' = $ExcludeFolders
|
|
'OfficeDocs' = $OfficeDocs
|
|
'ExcludeHidden' = $ExcludeHidden
|
|
'FreshEXEs' = $FreshEXEs
|
|
'CheckWriteAccess' = $CheckWriteAccess
|
|
'OutFile' = $OutFile
|
|
'UsePSDrive' = $UsePSDrive
|
|
}
|
|
|
|
# kick off the threaded script block + arguments
|
|
if($Shares) {
|
|
# pass the shares as the hosts so the threaded function code doesn't have to be hacked up
|
|
Invoke-ThreadedFunction -ComputerName $Shares -ScriptBlock $HostEnumBlock -ScriptParameters $ScriptParams -Threads $Threads
|
|
}
|
|
else {
|
|
Invoke-ThreadedFunction -ComputerName $ComputerName -ScriptBlock $HostEnumBlock -ScriptParameters $ScriptParams -Threads $Threads
|
|
}
|
|
}
|
|
|
|
else {
|
|
if($Shares){
|
|
$ComputerName = $Shares
|
|
}
|
|
elseif(-not $NoPing -and ($ComputerName.count -gt 1)) {
|
|
# ping all hosts in parallel
|
|
$Ping = {param($ComputerName) if(Test-Connection -ComputerName $ComputerName -Count 1 -Quiet -ErrorAction Stop){$ComputerName}}
|
|
$ComputerName = Invoke-ThreadedFunction -NoImports -ComputerName $ComputerName -ScriptBlock $Ping -Threads 100
|
|
}
|
|
|
|
Write-Verbose "[*] Total number of active hosts: $($ComputerName.count)"
|
|
$Counter = 0
|
|
|
|
$ComputerName | Where-Object {$_} | ForEach-Object {
|
|
Write-Verbose "Computer: $_"
|
|
$Counter = $Counter + 1
|
|
|
|
# sleep for our semi-randomized interval
|
|
Start-Sleep -Seconds $RandNo.Next((1-$Jitter)*$Delay, (1+$Jitter)*$Delay)
|
|
|
|
Write-Verbose "[*] Enumerating server $_ ($Counter of $($ComputerName.count))"
|
|
|
|
Invoke-Command -ScriptBlock $HostEnumBlock -ArgumentList $_, $False, $ExcludedShares, $SearchTerms, $ExcludeFolders, $OfficeDocs, $ExcludeHidden, $FreshEXEs, $CheckWriteAccess, $OutFile, $UsePSDrive
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
function Get-ThisThingDomainTrust {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0, ValueFromPipeline=$True)]
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[String]
|
|
$ADSpath,
|
|
|
|
[Switch]
|
|
$API,
|
|
|
|
[Switch]
|
|
$LDAP,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
begin {
|
|
$TrustAttributes = @{
|
|
[uint32]'0x00000001' = 'non_transitive'
|
|
[uint32]'0x00000002' = 'uplevel_only'
|
|
[uint32]'0x00000004' = 'quarantined_domain'
|
|
[uint32]'0x00000008' = 'forest_transitive'
|
|
[uint32]'0x00000010' = 'cross_organization'
|
|
[uint32]'0x00000020' = 'within_forest'
|
|
[uint32]'0x00000040' = 'treat_as_external'
|
|
[uint32]'0x00000080' = 'trust_uses_rc4_encryption'
|
|
[uint32]'0x00000100' = 'trust_uses_aes_keys'
|
|
[uint32]'0x00000200' = 'cross_organization_no_tgt_delegation'
|
|
[uint32]'0x00000400' = 'pim_trust'
|
|
}
|
|
}
|
|
|
|
process {
|
|
|
|
if(-not $Domain) {
|
|
# if not domain is specified grab the current domain
|
|
$SourceDomain = (Get-ThisThingDomain -Credential $Credential).Name
|
|
}
|
|
else {
|
|
$SourceDomain = $Domain
|
|
}
|
|
|
|
if($LDAP -or $ADSPath) {
|
|
|
|
$TrustSearcher = Get-DomainSearcher -Domain $SourceDomain -DomainController $DomainController -Credential $Credential -PageSize $PageSize -ADSpath $ADSpath
|
|
|
|
$SourceSID = Get-DomainSID -Domain $SourceDomain -DomainController $DomainController
|
|
|
|
if($TrustSearcher) {
|
|
|
|
$TrustSearcher.Filter = '(objectClass=trustedDomain)'
|
|
|
|
$Results = $TrustSearcher.FindAll()
|
|
$Results | Where-Object {$_} | ForEach-Object {
|
|
$Props = $_.Properties
|
|
$DomainTrust = New-Object PSObject
|
|
|
|
$TrustAttrib = @()
|
|
$TrustAttrib += $TrustAttributes.Keys | Where-Object { $Props.trustattributes[0] -band $_ } | ForEach-Object { $TrustAttributes[$_] }
|
|
|
|
$Direction = Switch ($Props.trustdirection) {
|
|
0 { 'Disabled' }
|
|
1 { 'Inbound' }
|
|
2 { 'Outbound' }
|
|
3 { 'Bidirectional' }
|
|
}
|
|
$ObjectGuid = New-Object Guid @(,$Props.objectguid[0])
|
|
$TargetSID = (New-Object System.Security.Principal.SecurityIdentifier($Props.securityidentifier[0],0)).Value
|
|
$DomainTrust | Add-Member Noteproperty 'SourceName' $SourceDomain
|
|
$DomainTrust | Add-Member Noteproperty 'SourceSID' $SourceSID
|
|
$DomainTrust | Add-Member Noteproperty 'TargetName' $Props.name[0]
|
|
$DomainTrust | Add-Member Noteproperty 'TargetSID' $TargetSID
|
|
$DomainTrust | Add-Member Noteproperty 'ObjectGuid' "{$ObjectGuid}"
|
|
$DomainTrust | Add-Member Noteproperty 'TrustType' $($TrustAttrib -join ',')
|
|
$DomainTrust | Add-Member Noteproperty 'TrustDirection' "$Direction"
|
|
$DomainTrust.PSObject.TypeNames.Add('PowerView.DomainTrustLDAP')
|
|
$DomainTrust
|
|
}
|
|
$Results.dispose()
|
|
$TrustSearcher.dispose()
|
|
}
|
|
}
|
|
elseif($API) {
|
|
if(-not $DomainController) {
|
|
$DomainController = Get-ThisThingDomainController -Credential $Credential -Domain $SourceDomain | Select-Object -First 1 | Select-Object -ExpandProperty Name
|
|
}
|
|
|
|
if($DomainController) {
|
|
# arguments for DsEnumerateDomainTrusts
|
|
$PtrInfo = [IntPtr]::Zero
|
|
|
|
# 63 = DS_DOMAIN_IN_FOREST + DS_DOMAIN_DIRECT_OUTBOUND + DS_DOMAIN_TREE_ROOT + DS_DOMAIN_PRIMARY + DS_DOMAIN_NATIVE_MODE + DS_DOMAIN_DIRECT_INBOUND
|
|
$Flags = 63
|
|
$DomainCount = 0
|
|
|
|
# get the trust information from the target server
|
|
$Result = $Netapi32::DsEnumerateDomainTrusts($DomainController, $Flags, [ref]$PtrInfo, [ref]$DomainCount)
|
|
|
|
# Locate the offset of the initial intPtr
|
|
$Offset = $PtrInfo.ToInt64()
|
|
|
|
# 0 = success
|
|
if (($Result -eq 0) -and ($Offset -gt 0)) {
|
|
|
|
# Work out how much to increment the pointer by finding out the size of the structure
|
|
$Increment = $DS_DOMAIN_TRUSTS::GetSize()
|
|
|
|
# parse all the result structures
|
|
for ($i = 0; ($i -lt $DomainCount); $i++) {
|
|
# create a new int ptr at the given offset and cast the pointer as our result structure
|
|
$NewIntPtr = New-Object System.Intptr -ArgumentList $Offset
|
|
$Info = $NewIntPtr -as $DS_DOMAIN_TRUSTS
|
|
|
|
$Offset = $NewIntPtr.ToInt64()
|
|
$Offset += $Increment
|
|
|
|
$SidString = ""
|
|
$Result = $Advapi32::ConvertSidToStringSid($Info.DomainSid, [ref]$SidString);$LastError = [Runtime.InteropServices.Marshal]::GetLastWin32Error()
|
|
|
|
if($Result -eq 0) {
|
|
Write-Verbose "Error: $(([ComponentModel.Win32Exception] $LastError).Message)"
|
|
}
|
|
else {
|
|
$DomainTrust = New-Object PSObject
|
|
$DomainTrust | Add-Member Noteproperty 'SourceDomain' $SourceDomain
|
|
$DomainTrust | Add-Member Noteproperty 'SourceDomainController' $DomainController
|
|
$DomainTrust | Add-Member Noteproperty 'NetbiosDomainName' $Info.NetbiosDomainName
|
|
$DomainTrust | Add-Member Noteproperty 'DnsDomainName' $Info.DnsDomainName
|
|
$DomainTrust | Add-Member Noteproperty 'Flags' $Info.Flags
|
|
$DomainTrust | Add-Member Noteproperty 'ParentIndex' $Info.ParentIndex
|
|
$DomainTrust | Add-Member Noteproperty 'TrustType' $Info.TrustType
|
|
$DomainTrust | Add-Member Noteproperty 'TrustAttributes' $Info.TrustAttributes
|
|
$DomainTrust | Add-Member Noteproperty 'DomainSid' $SidString
|
|
$DomainTrust | Add-Member Noteproperty 'DomainGuid' $Info.DomainGuid
|
|
$DomainTrust.PSObject.TypeNames.Add('PowerView.APIDomainTrust')
|
|
$DomainTrust
|
|
}
|
|
}
|
|
# free up the result buffer
|
|
$Null = $Netapi32::NetApiBufferFree($PtrInfo)
|
|
}
|
|
else {
|
|
Write-Verbose "Error: $(([ComponentModel.Win32Exception] $Result).Message)"
|
|
}
|
|
}
|
|
else {
|
|
Write-Verbose "Could not retrieve domain controller for $Domain"
|
|
}
|
|
}
|
|
else {
|
|
# if we're using direct domain connections through .NET
|
|
$FoundDomain = Get-ThisThingDomain -Domain $Domain -Credential $Credential
|
|
if($FoundDomain) {
|
|
$FoundDomain.GetAllTrustRelationships() | ForEach-Object {
|
|
$_.PSObject.TypeNames.Add('PowerView.DomainTrust')
|
|
$_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
function Get-ThisThingForestTrust {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Position=0,ValueFromPipeline=$True)]
|
|
[String]
|
|
$Forest,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
process {
|
|
$FoundForest = Get-ThisThingForest -Forest $Forest -Credential $Credential
|
|
|
|
if($FoundForest) {
|
|
$FoundForest.GetAllTrustRelationships() | ForEach-Object {
|
|
$_.PSObject.TypeNames.Add('PowerView.ForestTrust')
|
|
$_
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
function Find-ForeignUser {
|
|
[CmdletBinding()]
|
|
param(
|
|
[String]
|
|
$UserName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$LDAP,
|
|
|
|
[Switch]
|
|
$Recurse,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
function Get-ForeignUser {
|
|
# helper used to enumerate users who are in groups outside of their principal domain
|
|
param(
|
|
[String]
|
|
$UserName,
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
if ($Domain) {
|
|
# get the domain name into distinguished form
|
|
$DistinguishedDomainName = "DC=" + $Domain -replace '\.',',DC='
|
|
}
|
|
else {
|
|
$DistinguishedDomainName = [String] ([adsi]'').distinguishedname
|
|
$Domain = $DistinguishedDomainName -replace 'DC=','' -replace ',','.'
|
|
}
|
|
|
|
Get-ThisThingUser -Domain $Domain -DomainController $DomainController -UserName $UserName -PageSize $PageSize -Filter '(memberof=*)' | ForEach-Object {
|
|
ForEach ($Membership in $_.memberof) {
|
|
$Index = $Membership.IndexOf("DC=")
|
|
if($Index) {
|
|
|
|
$GroupDomain = $($Membership.substring($Index)) -replace 'DC=','' -replace ',','.'
|
|
|
|
if ($GroupDomain.CompareTo($Domain)) {
|
|
# if the group domain doesn't match the user domain, output
|
|
$GroupName = $Membership.split(",")[0].split("=")[1]
|
|
$ForeignUser = New-Object PSObject
|
|
$ForeignUser | Add-Member Noteproperty 'UserDomain' $Domain
|
|
$ForeignUser | Add-Member Noteproperty 'UserName' $_.samaccountname
|
|
$ForeignUser | Add-Member Noteproperty 'GroupDomain' $GroupDomain
|
|
$ForeignUser | Add-Member Noteproperty 'GroupName' $GroupName
|
|
$ForeignUser | Add-Member Noteproperty 'GroupDN' $Membership
|
|
$ForeignUser
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($Recurse) {
|
|
# get all rechable domains in the trust mesh and uniquify them
|
|
if($LDAP -or $DomainController) {
|
|
$DomainTrusts = Invoke-MapDomainTrust -LDAP -DomainController $DomainController -PageSize $PageSize | ForEach-Object { $_.SourceDomain } | Sort-Object -Unique
|
|
}
|
|
else {
|
|
$DomainTrusts = Invoke-MapDomainTrust -PageSize $PageSize | ForEach-Object { $_.SourceDomain } | Sort-Object -Unique
|
|
}
|
|
|
|
ForEach($DomainTrust in $DomainTrusts) {
|
|
# get the trust groups for each domain in the trust mesh
|
|
Write-Verbose "Enumerating trust groups in domain $DomainTrust"
|
|
Get-ForeignUser -Domain $DomainTrust -UserName $UserName -PageSize $PageSize
|
|
}
|
|
}
|
|
else {
|
|
Get-ForeignUser -Domain $Domain -DomainController $DomainController -UserName $UserName -PageSize $PageSize
|
|
}
|
|
}
|
|
|
|
|
|
function Find-ForeignGroup {
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[String]
|
|
$GroupName = '*',
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[Switch]
|
|
$LDAP,
|
|
|
|
[Switch]
|
|
$Recurse,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
function Get-ForeignGroup {
|
|
param(
|
|
[String]
|
|
$GroupName = '*',
|
|
|
|
[String]
|
|
$Domain,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200
|
|
)
|
|
|
|
if(-not $Domain) {
|
|
$Domain = (Get-ThisThingDomain).Name
|
|
}
|
|
|
|
$DomainDN = "DC=$($Domain.Replace('.', ',DC='))"
|
|
Write-Verbose "DomainDN: $DomainDN"
|
|
|
|
# standard group names to ignore
|
|
$ExcludeGroups = @("Users", "Domain Users", "Guests")
|
|
|
|
# get all the groupnames for the given domain
|
|
Get-ThisThingGroup -GroupName $GroupName -Filter '(member=*)' -Domain $Domain -DomainController $DomainController -FullData -PageSize $PageSize | Where-Object {
|
|
# exclude common large groups
|
|
-not ($ExcludeGroups -contains $_.samaccountname) } | ForEach-Object {
|
|
|
|
$GroupName = $_.samAccountName
|
|
|
|
$_.member | ForEach-Object {
|
|
# filter for foreign SIDs in the cn field for users in another domain,
|
|
# or if the DN doesn't end with the proper DN for the queried domain
|
|
if (($_ -match 'CN=S-1-5-21.*-.*') -or ($DomainDN -ne ($_.substring($_.IndexOf("DC="))))) {
|
|
|
|
$UserDomain = $_.subString($_.IndexOf("DC=")) -replace 'DC=','' -replace ',','.'
|
|
$UserName = $_.split(",")[0].split("=")[1]
|
|
|
|
$ForeignGroupUser = New-Object PSObject
|
|
$ForeignGroupUser | Add-Member Noteproperty 'GroupDomain' $Domain
|
|
$ForeignGroupUser | Add-Member Noteproperty 'GroupName' $GroupName
|
|
$ForeignGroupUser | Add-Member Noteproperty 'UserDomain' $UserDomain
|
|
$ForeignGroupUser | Add-Member Noteproperty 'UserName' $UserName
|
|
$ForeignGroupUser | Add-Member Noteproperty 'UserDN' $_
|
|
$ForeignGroupUser
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($Recurse) {
|
|
# get all rechable domains in the trust mesh and uniquify them
|
|
if($LDAP -or $DomainController) {
|
|
$DomainTrusts = Invoke-MapDomainTrust -LDAP -DomainController $DomainController -PageSize $PageSize | ForEach-Object { $_.SourceDomain } | Sort-Object -Unique
|
|
}
|
|
else {
|
|
$DomainTrusts = Invoke-MapDomainTrust -PageSize $PageSize | ForEach-Object { $_.SourceDomain } | Sort-Object -Unique
|
|
}
|
|
|
|
ForEach($DomainTrust in $DomainTrusts) {
|
|
# get the trust groups for each domain in the trust mesh
|
|
Write-Verbose "Enumerating trust groups in domain $DomainTrust"
|
|
Get-ForeignGroup -GroupName $GroupName -Domain $Domain -DomainController $DomainController -PageSize $PageSize
|
|
}
|
|
}
|
|
else {
|
|
Get-ForeignGroup -GroupName $GroupName -Domain $Domain -DomainController $DomainController -PageSize $PageSize
|
|
}
|
|
}
|
|
|
|
|
|
function Find-ManagedSecurityGroups {
|
|
|
|
# Go through the list of security groups on the domain and identify those who have a manager
|
|
Get-ThisThingGroup -FullData -Filter '(managedBy=*)' | Select-Object -Unique distinguishedName,managedBy,cn | ForEach-Object {
|
|
|
|
# Retrieve the object that the managedBy DN refers to
|
|
$group_manager = Get-ADObject -ADSPath $_.managedBy | Select-Object cn,distinguishedname,name,samaccounttype,samaccountname
|
|
|
|
# Create a results object to store our findings
|
|
$results_object = New-Object -TypeName PSObject -Property @{
|
|
'GroupCN' = $_.cn
|
|
'GroupDN' = $_.distinguishedname
|
|
'ManagerCN' = $group_manager.cn
|
|
'ManagerDN' = $group_manager.distinguishedName
|
|
'ManagerSAN' = $group_manager.samaccountname
|
|
'ManagerType' = ''
|
|
'CanManagerWrite' = $FALSE
|
|
}
|
|
|
|
# Determine whether the manager is a user or a group
|
|
if ($group_manager.samaccounttype -eq 0x10000000) {
|
|
$results_object.ManagerType = 'Group'
|
|
} elseif ($group_manager.samaccounttype -eq 0x30000000) {
|
|
$results_object.ManagerType = 'User'
|
|
}
|
|
|
|
# Find the ACLs that relate to the ability to write to the group
|
|
$xacl = Get-ObjectAcl -ADSPath $_.distinguishedname -Rights WriteMembers
|
|
|
|
# Double-check that the manager
|
|
if ($xacl.ObjectType -eq 'bf9679c0-0de6-11d0-a285-00aa003049e2' -and $xacl.AccessControlType -eq 'Allow' -and $xacl.IdentityReference.Value.Contains($group_manager.samaccountname)) {
|
|
$results_object.CanManagerWrite = $TRUE
|
|
}
|
|
$results_object
|
|
}
|
|
}
|
|
|
|
|
|
function Invoke-MapDomainTrust {
|
|
[CmdletBinding()]
|
|
param(
|
|
[Switch]
|
|
$LDAP,
|
|
|
|
[String]
|
|
$DomainController,
|
|
|
|
[ValidateRange(1,10000)]
|
|
[Int]
|
|
$PageSize = 200,
|
|
|
|
[Management.Automation.PSCredential]
|
|
$Credential
|
|
)
|
|
|
|
# keep track of domains seen so we don't hit infinite recursion
|
|
$SeenDomains = @{}
|
|
|
|
# our domain status tracker
|
|
$Domains = New-Object System.Collections.Stack
|
|
|
|
# get the current domain and push it onto the stack
|
|
$CurrentDomain = (Get-ThisThingDomain -Credential $Credential).Name
|
|
$Domains.push($CurrentDomain)
|
|
|
|
while($Domains.Count -ne 0) {
|
|
|
|
$Domain = $Domains.Pop()
|
|
|
|
# if we haven't seen this domain before
|
|
if ($Domain -and ($Domain.Trim() -ne "") -and (-not $SeenDomains.ContainsKey($Domain))) {
|
|
|
|
Write-Verbose "Enumerating trusts for domain '$Domain'"
|
|
|
|
# mark it as seen in our list
|
|
$Null = $SeenDomains.add($Domain, "")
|
|
|
|
try {
|
|
# get all the trusts for this domain
|
|
if($LDAP -or $DomainController) {
|
|
$Trusts = Get-ThisThingDomainTrust -Domain $Domain -LDAP -DomainController $DomainController -PageSize $PageSize -Credential $Credential
|
|
}
|
|
else {
|
|
$Trusts = Get-ThisThingDomainTrust -Domain $Domain -PageSize $PageSize -Credential $Credential
|
|
}
|
|
|
|
if($Trusts -isnot [System.Array]) {
|
|
$Trusts = @($Trusts)
|
|
}
|
|
|
|
# get any forest trusts, if they exist
|
|
if(-not ($LDAP -or $DomainController) ) {
|
|
$Trusts += Get-ThisThingForestTrust -Forest $Domain -Credential $Credential
|
|
}
|
|
|
|
if ($Trusts) {
|
|
if($Trusts -isnot [System.Array]) {
|
|
$Trusts = @($Trusts)
|
|
}
|
|
|
|
# enumerate each trust found
|
|
ForEach ($Trust in $Trusts) {
|
|
if($Trust.SourceName -and $Trust.TargetName) {
|
|
$SourceDomain = $Trust.SourceName
|
|
$TargetDomain = $Trust.TargetName
|
|
$TrustType = $Trust.TrustType
|
|
$TrustDirection = $Trust.TrustDirection
|
|
$ObjectType = $Trust.PSObject.TypeNames | Where-Object {$_ -match 'PowerView'} | Select-Object -First 1
|
|
|
|
# make sure we process the target
|
|
$Null = $Domains.Push($TargetDomain)
|
|
|
|
# build the nicely-parsable custom output object
|
|
$DomainTrust = New-Object PSObject
|
|
$DomainTrust | Add-Member Noteproperty 'SourceDomain' "$SourceDomain"
|
|
$DomainTrust | Add-Member Noteproperty 'SourceSID' $Trust.SourceSID
|
|
$DomainTrust | Add-Member Noteproperty 'TargetDomain' "$TargetDomain"
|
|
$DomainTrust | Add-Member Noteproperty 'TargetSID' $Trust.TargetSID
|
|
$DomainTrust | Add-Member Noteproperty 'TrustType' "$TrustType"
|
|
$DomainTrust | Add-Member Noteproperty 'TrustDirection' "$TrustDirection"
|
|
$DomainTrust.PSObject.TypeNames.Add($ObjectType)
|
|
$DomainTrust
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Verbose "[!] Error: $_"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$Mod = New-InMemoryModule -ModuleName Win32
|
|
|
|
$FunctionDefinitions = @(
|
|
(func netapi32 NetShareEnum ([Int]) @([String], [Int], [IntPtr].MakeByRefType(), [Int], [Int32].MakeByRefType(), [Int32].MakeByRefType(), [Int32].MakeByRefType())),
|
|
(func netapi32 NetWkstaUserEnum ([Int]) @([String], [Int], [IntPtr].MakeByRefType(), [Int], [Int32].MakeByRefType(), [Int32].MakeByRefType(), [Int32].MakeByRefType())),
|
|
(func netapi32 NetSessionEnum ([Int]) @([String], [String], [String], [Int], [IntPtr].MakeByRefType(), [Int], [Int32].MakeByRefType(), [Int32].MakeByRefType(), [Int32].MakeByRefType())),
|
|
(func netapi32 NetLocalGroupGetMembers ([Int]) @([String], [String], [Int], [IntPtr].MakeByRefType(), [Int], [Int32].MakeByRefType(), [Int32].MakeByRefType(), [Int32].MakeByRefType())),
|
|
(func netapi32 DsGetSiteName ([Int]) @([String], [IntPtr].MakeByRefType())),
|
|
(func netapi32 DsEnumerateDomainTrusts ([Int]) @([String], [UInt32], [IntPtr].MakeByRefType(), [IntPtr].MakeByRefType())),
|
|
(func netapi32 NetApiBufferFree ([Int]) @([IntPtr])),
|
|
(func advapi32 ConvertSidToStringSid ([Int]) @([IntPtr], [String].MakeByRefType()) -SetLastError),
|
|
(func advapi32 OpenSCManagerW ([IntPtr]) @([String], [String], [Int]) -SetLastError),
|
|
(func advapi32 CloseServiceHandle ([Int]) @([IntPtr])),
|
|
(func wtsapi32 WTSOpenServerEx ([IntPtr]) @([String])),
|
|
(func wtsapi32 WTSEnumerateSessionsEx ([Int]) @([IntPtr], [Int32].MakeByRefType(), [Int], [IntPtr].MakeByRefType(), [Int32].MakeByRefType()) -SetLastError),
|
|
(func wtsapi32 WTSQuerySessionInformation ([Int]) @([IntPtr], [Int], [Int], [IntPtr].MakeByRefType(), [Int32].MakeByRefType()) -SetLastError),
|
|
(func wtsapi32 WTSFreeMemoryEx ([Int]) @([Int32], [IntPtr], [Int32])),
|
|
(func wtsapi32 WTSFreeMemory ([Int]) @([IntPtr])),
|
|
(func wtsapi32 WTSCloseServer ([Int]) @([IntPtr]))
|
|
)
|
|
|
|
# enum used by $WTS_SESSION_INFO_1 below
|
|
$WTSConnectState = psenum $Mod WTS_CONNECTSTATE_CLASS UInt16 @{
|
|
Active = 0
|
|
Connected = 1
|
|
ConnectQuery = 2
|
|
Shadow = 3
|
|
Disconnected = 4
|
|
Idle = 5
|
|
Listen = 6
|
|
Reset = 7
|
|
Down = 8
|
|
Init = 9
|
|
}
|
|
|
|
# the WTSEnumerateSessionsEx result structure
|
|
$WTS_SESSION_INFO_1 = struct $Mod WTS_SESSION_INFO_1 @{
|
|
ExecEnvId = field 0 UInt32
|
|
State = field 1 $WTSConnectState
|
|
SessionId = field 2 UInt32
|
|
pSessionName = field 3 String -MarshalAs @('LPWStr')
|
|
pHostName = field 4 String -MarshalAs @('LPWStr')
|
|
pUserName = field 5 String -MarshalAs @('LPWStr')
|
|
pDomainName = field 6 String -MarshalAs @('LPWStr')
|
|
pFarmName = field 7 String -MarshalAs @('LPWStr')
|
|
}
|
|
|
|
# the particular WTSQuerySessionInformation result structure
|
|
$WTS_CLIENT_ADDRESS = struct $mod WTS_CLIENT_ADDRESS @{
|
|
AddressFamily = field 0 UInt32
|
|
Address = field 1 Byte[] -MarshalAs @('ByValArray', 20)
|
|
}
|
|
|
|
# the NetShareEnum result structure
|
|
$SHARE_INFO_1 = struct $Mod SHARE_INFO_1 @{
|
|
shi1_netname = field 0 String -MarshalAs @('LPWStr')
|
|
shi1_type = field 1 UInt32
|
|
shi1_remark = field 2 String -MarshalAs @('LPWStr')
|
|
}
|
|
|
|
# the NetWkstaUserEnum result structure
|
|
$WKSTA_USER_INFO_1 = struct $Mod WKSTA_USER_INFO_1 @{
|
|
wkui1_username = field 0 String -MarshalAs @('LPWStr')
|
|
wkui1_logon_domain = field 1 String -MarshalAs @('LPWStr')
|
|
wkui1_oth_domains = field 2 String -MarshalAs @('LPWStr')
|
|
wkui1_logon_server = field 3 String -MarshalAs @('LPWStr')
|
|
}
|
|
|
|
# the NetSessionEnum result structure
|
|
$SESSION_INFO_10 = struct $Mod SESSION_INFO_10 @{
|
|
sesi10_cname = field 0 String -MarshalAs @('LPWStr')
|
|
sesi10_username = field 1 String -MarshalAs @('LPWStr')
|
|
sesi10_time = field 2 UInt32
|
|
sesi10_idle_time = field 3 UInt32
|
|
}
|
|
|
|
# enum used by $LOCALGROUP_MEMBERS_INFO_2 below
|
|
$SID_NAME_USE = psenum $Mod SID_NAME_USE UInt16 @{
|
|
SidTypeUser = 1
|
|
SidTypeGroup = 2
|
|
SidTypeDomain = 3
|
|
SidTypeAlias = 4
|
|
SidTypeWellKnownGroup = 5
|
|
SidTypeDeletedAccount = 6
|
|
SidTypeInvalid = 7
|
|
SidTypeUnknown = 8
|
|
SidTypeComputer = 9
|
|
}
|
|
|
|
# the NetLocalGroupGetMembers result structure
|
|
$LOCALGROUP_MEMBERS_INFO_2 = struct $Mod LOCALGROUP_MEMBERS_INFO_2 @{
|
|
lgrmi2_sid = field 0 IntPtr
|
|
lgrmi2_sidusage = field 1 $SID_NAME_USE
|
|
lgrmi2_domainandname = field 2 String -MarshalAs @('LPWStr')
|
|
}
|
|
|
|
# enums used in DS_DOMAIN_TRUSTS
|
|
$DsDomainFlag = psenum $Mod DsDomain.Flags UInt32 @{
|
|
IN_FOREST = 1
|
|
DIRECT_OUTBOUND = 2
|
|
TREE_ROOT = 4
|
|
PRIMARY = 8
|
|
NATIVE_MODE = 16
|
|
DIRECT_INBOUND = 32
|
|
} -Bitfield
|
|
$DsDomainTrustType = psenum $Mod DsDomain.TrustType UInt32 @{
|
|
DOWNLEVEL = 1
|
|
UPLEVEL = 2
|
|
MIT = 3
|
|
DCE = 4
|
|
}
|
|
$DsDomainTrustAttributes = psenum $Mod DsDomain.TrustAttributes UInt32 @{
|
|
NON_TRANSITIVE = 1
|
|
UPLEVEL_ONLY = 2
|
|
FILTER_SIDS = 4
|
|
FOREST_TRANSITIVE = 8
|
|
CROSS_ORGANIZATION = 16
|
|
WITHIN_FOREST = 32
|
|
TREAT_AS_EXTERNAL = 64
|
|
}
|
|
|
|
# the DsEnumerateDomainTrusts result structure
|
|
$DS_DOMAIN_TRUSTS = struct $Mod DS_DOMAIN_TRUSTS @{
|
|
NetbiosDomainName = field 0 String -MarshalAs @('LPWStr')
|
|
DnsDomainName = field 1 String -MarshalAs @('LPWStr')
|
|
Flags = field 2 $DsDomainFlag
|
|
ParentIndex = field 3 UInt32
|
|
TrustType = field 4 $DsDomainTrustType
|
|
TrustAttributes = field 5 $DsDomainTrustAttributes
|
|
DomainSid = field 6 IntPtr
|
|
DomainGuid = field 7 Guid
|
|
}
|
|
|
|
$Types = $FunctionDefinitions | Add-Win32Type -Module $Mod -Namespace 'Win32'
|
|
$Netapi32 = $Types['netapi32']
|
|
$Advapi32 = $Types['advapi32']
|
|
$Wtsapi32 = $Types['wtsapi32']
|
|
|
|
function Invoke-Parallel
|
|
{
|
|
[cmdletbinding(DefaultParameterSetName = 'ScriptBlock')]
|
|
Param (
|
|
[Parameter(Mandatory = $false,position = 0,ParameterSetName = 'ScriptBlock')]
|
|
[System.Management.Automation.ScriptBlock]$ScriptBlock,
|
|
|
|
[Parameter(Mandatory = $false,ParameterSetName = 'ScriptFile')]
|
|
[ValidateScript({
|
|
Test-Path $_ -PathType leaf
|
|
})]
|
|
$ScriptFile,
|
|
|
|
[Parameter(Mandatory = $true,ValueFromPipeline = $true)]
|
|
[Alias('CN','__Server','IPAddress','Server','ComputerName')]
|
|
[PSObject]$InputObject,
|
|
|
|
[PSObject]$Parameter,
|
|
|
|
[switch]$ImportSessionFunctions,
|
|
|
|
[switch]$ImportVariables,
|
|
|
|
[switch]$ImportModules,
|
|
|
|
[int]$Throttle = 20,
|
|
|
|
[int]$SleepTimer = 200,
|
|
|
|
[int]$RunspaceTimeout = 0,
|
|
|
|
[switch]$NoCloseOnTimeout = $false,
|
|
|
|
[int]$MaxQueue,
|
|
|
|
[validatescript({
|
|
Test-Path (Split-Path -Path $_ -Parent)
|
|
})]
|
|
[string]$LogFile = 'C:\temp\log.log',
|
|
|
|
[switch] $Quiet = $false
|
|
)
|
|
|
|
Begin {
|
|
|
|
#No max queue specified? Estimate one.
|
|
#We use the script scope to resolve an odd PowerShell 2 issue where MaxQueue isn't seen later in the function
|
|
if( -not $PSBoundParameters.ContainsKey('MaxQueue') )
|
|
{
|
|
if($RunspaceTimeout -ne 0)
|
|
{
|
|
$script:MaxQueue = $Throttle
|
|
}
|
|
else
|
|
{
|
|
$script:MaxQueue = $Throttle * 3
|
|
}
|
|
}
|
|
else
|
|
{
|
|
$script:MaxQueue = $MaxQueue
|
|
}
|
|
|
|
#If they want to import variables or modules, create a clean runspace, get loaded items, use those to exclude items
|
|
if ($ImportVariables -or $ImportModules)
|
|
{
|
|
$StandardUserEnv = [powershell]::Create().addscript({
|
|
#Get modules and snapins in this clean runspace
|
|
$Modules = Get-Module | Select-Object -ExpandProperty Name
|
|
$Snapins = Get-PSSnapin | Select-Object -ExpandProperty Name
|
|
|
|
#Get variables in this clean runspace
|
|
#Called last to get vars like $? into session
|
|
$Variables = Get-Variable | Select-Object -ExpandProperty Name
|
|
|
|
#Return a hashtable where we can access each.
|
|
@{
|
|
Variables = $Variables
|
|
Modules = $Modules
|
|
Snapins = $Snapins
|
|
}
|
|
}).invoke()[0]
|
|
|
|
if ($ImportVariables)
|
|
{
|
|
#Exclude common parameters, bound parameters, and automatic variables
|
|
Function _temp
|
|
{
|
|
[cmdletbinding()] param()
|
|
}
|
|
$VariablesToExclude = @( (Get-Command _temp | Select-Object -ExpandProperty parameters).Keys + $PSBoundParameters.Keys + $StandardUserEnv.Variables )
|
|
#Write-Verbose "Excluding variables $( ($VariablesToExclude | sort ) -join ", ")"
|
|
|
|
# we don't use 'Get-Variable -Exclude', because it uses regexps.
|
|
# One of the veriables that we pass is '$?'.
|
|
# There could be other variables with such problems.
|
|
# Scope 2 required if we move to a real module
|
|
$UserVariables = @( Get-Variable | Where-Object -FilterScript {
|
|
-not ($VariablesToExclude -contains $_.Name)
|
|
} )
|
|
#Write-Verbose "Found variables to import: $( ($UserVariables | Select -expandproperty Name | Sort ) -join ", " | Out-String).`n"
|
|
}
|
|
|
|
if ($ImportModules)
|
|
{
|
|
$UserModules = @( Get-Module |
|
|
Where-Object -FilterScript {
|
|
$StandardUserEnv.Modules -notcontains $_.Name -and (Test-Path -Path $_.Path -ErrorAction SilentlyContinue)
|
|
} |
|
|
Select-Object -ExpandProperty Path )
|
|
$UserSnapins = @( Get-PSSnapin |
|
|
Select-Object -ExpandProperty Name |
|
|
Where-Object -FilterScript {
|
|
$StandardUserEnv.Snapins -notcontains $_
|
|
} )
|
|
}
|
|
}
|
|
|
|
#region functions
|
|
|
|
Function Get-RunspaceData
|
|
{
|
|
[cmdletbinding()]
|
|
param( [switch]$Wait )
|
|
|
|
#loop through runspaces
|
|
#if $wait is specified, keep looping until all complete
|
|
Do
|
|
{
|
|
#set more to false for tracking completion
|
|
$more = $false
|
|
|
|
#Progress bar if we have inputobject count (bound parameter)
|
|
if (-not $Quiet)
|
|
{
|
|
Write-Progress -Activity 'Running Query' -Status 'Starting threads'`
|
|
-CurrentOperation "$startedCount threads defined - $totalCount input objects - $script:completedCount input objects processed"`
|
|
-PercentComplete $( Try
|
|
{
|
|
$script:completedCount / $totalCount * 100
|
|
}
|
|
Catch
|
|
{
|
|
0
|
|
}
|
|
)
|
|
}
|
|
|
|
#run through each runspace.
|
|
Foreach($runspace in $runspaces)
|
|
{
|
|
#get the duration - inaccurate
|
|
$currentdate = Get-Date
|
|
$runtime = $currentdate - $runspace.startTime
|
|
$runMin = [math]::Round( $runtime.totalminutes ,2 )
|
|
|
|
#set up log object
|
|
$log = '' | Select-Object -Property Date, Action, Runtime, Status, Details
|
|
$log.Action = "Removing:'$($runspace.object)'"
|
|
$log.Date = $currentdate
|
|
$log.Runtime = "$runMin minutes"
|
|
|
|
#If runspace completed, end invoke, dispose, recycle, counter++
|
|
If ($runspace.Runspace.isCompleted)
|
|
{
|
|
$script:completedCount++
|
|
|
|
#check if there were errors
|
|
if($runspace.powershell.Streams.Error.Count -gt 0)
|
|
{
|
|
#set the logging info and move the file to completed
|
|
$log.status = 'CompletedWithErrors'
|
|
#Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
foreach($ErrorRecord in $runspace.powershell.Streams.Error)
|
|
{
|
|
Write-Error -ErrorRecord $ErrorRecord
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#add logging details and cleanup
|
|
$log.status = 'Completed'
|
|
#Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
}
|
|
|
|
#everything is logged, clean up the runspace
|
|
$runspace.powershell.EndInvoke($runspace.Runspace)
|
|
$runspace.powershell.dispose()
|
|
$runspace.Runspace = $null
|
|
$runspace.powershell = $null
|
|
}
|
|
|
|
#If runtime exceeds max, dispose the runspace
|
|
ElseIf ( $RunspaceTimeout -ne 0 -and $runtime.totalseconds -gt $RunspaceTimeout)
|
|
{
|
|
$script:completedCount++
|
|
$timedOutTasks = $true
|
|
|
|
#add logging details and cleanup
|
|
$log.status = 'TimedOut'
|
|
#Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
Write-Error -Message "Runspace timed out at $($runtime.totalseconds) seconds for the object:`n$($runspace.object | Out-String)"
|
|
|
|
#Depending on how it hangs, we could still get stuck here as dispose calls a synchronous method on the powershell instance
|
|
if (!$NoCloseOnTimeout)
|
|
{
|
|
$runspace.powershell.dispose()
|
|
}
|
|
$runspace.Runspace = $null
|
|
$runspace.powershell = $null
|
|
$completedCount++
|
|
}
|
|
|
|
#If runspace isn't null set more to true
|
|
ElseIf ($runspace.Runspace -ne $null )
|
|
{
|
|
$log = $null
|
|
$more = $true
|
|
}
|
|
}
|
|
|
|
#Clean out unused runspace jobs
|
|
$temphash = $runspaces.clone()
|
|
$temphash |
|
|
Where-Object -FilterScript {
|
|
$_.runspace -eq $null
|
|
} |
|
|
ForEach-Object -Process {
|
|
$runspaces.remove($_)
|
|
}
|
|
|
|
#sleep for a bit if we will loop again
|
|
if($PSBoundParameters['Wait'])
|
|
{
|
|
Start-Sleep -Milliseconds $SleepTimer
|
|
}
|
|
|
|
#Loop again only if -wait parameter and there are more runspaces to process
|
|
}
|
|
while ($more -and $PSBoundParameters['Wait'])
|
|
|
|
#End of runspace function
|
|
}
|
|
|
|
#endregion functions
|
|
|
|
#region Init
|
|
|
|
if($PSCmdlet.ParameterSetName -eq 'ScriptFile')
|
|
{
|
|
$ScriptBlock = [scriptblock]::Create( $(Get-Content $ScriptFile | Out-String) )
|
|
}
|
|
elseif($PSCmdlet.ParameterSetName -eq 'ScriptBlock')
|
|
{
|
|
#Start building parameter names for the param block
|
|
[string[]]$ParamsToAdd = '$_'
|
|
if( $PSBoundParameters.ContainsKey('Parameter') )
|
|
{
|
|
$ParamsToAdd += '$Parameter'
|
|
}
|
|
|
|
$UsingVariableData = $null
|
|
|
|
|
|
# This code enables $Using support through the AST.
|
|
# This is entirely from Boe Prox, and his https://github.com/proxb/PoshRSJob module; all credit to Boe!
|
|
|
|
if($PSVersionTable.PSVersion.Major -gt 2)
|
|
{
|
|
#Extract using references
|
|
$UsingVariables = $ScriptBlock.ast.FindAll({
|
|
$args[0] -is [System.Management.Automation.Language.UsingExpressionAst]
|
|
},$true)
|
|
|
|
If ($UsingVariables)
|
|
{
|
|
$List = New-Object -TypeName 'System.Collections.Generic.List`1[System.Management.Automation.Language.VariableExpressionAst]'
|
|
ForEach ($Ast in $UsingVariables)
|
|
{
|
|
[void]$List.Add($Ast.SubExpression)
|
|
}
|
|
|
|
$UsingVar = $UsingVariables |
|
|
Group-Object -Property SubExpression |
|
|
ForEach-Object -Process {
|
|
$_.Group |
|
|
Select-Object -First 1
|
|
}
|
|
|
|
#Extract the name, value, and create replacements for each
|
|
$UsingVariableData = ForEach ($Var in $UsingVar)
|
|
{
|
|
Try
|
|
{
|
|
$Value = Get-Variable -Name $Var.SubExpression.VariablePath.UserPath -ErrorAction Stop
|
|
[pscustomobject]@{
|
|
Name = $Var.SubExpression.Extent.Text
|
|
Value = $Value.Value
|
|
NewName = ('$__using_{0}' -f $Var.SubExpression.VariablePath.UserPath)
|
|
NewVarName = ('__using_{0}' -f $Var.SubExpression.VariablePath.UserPath)
|
|
}
|
|
}
|
|
Catch
|
|
{
|
|
Write-Error -Message "$($Var.SubExpression.Extent.Text) is not a valid Using: variable!"
|
|
}
|
|
}
|
|
$ParamsToAdd += $UsingVariableData | Select-Object -ExpandProperty NewName -Unique
|
|
|
|
$NewParams = $UsingVariableData.NewName -join ', '
|
|
$Tuple = [Tuple]::Create($List, $NewParams)
|
|
$bindingFlags = [Reflection.BindingFlags]'Default,NonPublic,Instance'
|
|
$GetWithInputHandlingForInvokeCommandImpl = ($ScriptBlock.ast.gettype().GetMethod('GetWithInputHandlingForInvokeCommandImpl',$bindingFlags))
|
|
|
|
$StringScriptBlock = $GetWithInputHandlingForInvokeCommandImpl.Invoke($ScriptBlock.ast,@($Tuple))
|
|
|
|
$ScriptBlock = [scriptblock]::Create($StringScriptBlock)
|
|
|
|
#Write-Verbose $StringScriptBlock
|
|
}
|
|
}
|
|
|
|
$ScriptBlock = $ExecutionContext.InvokeCommand.NewScriptBlock("param($($ParamsToAdd -Join ', '))`r`n" + $ScriptBlock.ToString())
|
|
}
|
|
else
|
|
{
|
|
Throw 'Must provide ScriptBlock or ScriptFile'
|
|
Break
|
|
}
|
|
|
|
Write-Debug -Message "`$ScriptBlock: $($ScriptBlock | Out-String)"
|
|
If (-not($SuppressVerbose)){
|
|
Write-Verbose -Message 'Creating runspace pool and session states'
|
|
}
|
|
|
|
|
|
#If specified, add variables and modules/snapins to session state
|
|
$sessionstate = [System.Management.Automation.Runspaces.InitialSessionState]::CreateDefault()
|
|
if ($ImportVariables)
|
|
{
|
|
if($UserVariables.count -gt 0)
|
|
{
|
|
foreach($Variable in $UserVariables)
|
|
{
|
|
$sessionstate.Variables.Add( (New-Object -TypeName System.Management.Automation.Runspaces.SessionStateVariableEntry -ArgumentList $Variable.Name, $Variable.Value, $null) )
|
|
}
|
|
}
|
|
}
|
|
if ($ImportModules)
|
|
{
|
|
if($UserModules.count -gt 0)
|
|
{
|
|
foreach($ModulePath in $UserModules)
|
|
{
|
|
$sessionstate.ImportPSModule($ModulePath)
|
|
}
|
|
}
|
|
if($UserSnapins.count -gt 0)
|
|
{
|
|
foreach($PSSnapin in $UserSnapins)
|
|
{
|
|
[void]$sessionstate.ImportPSSnapIn($PSSnapin, [ref]$null)
|
|
}
|
|
}
|
|
}
|
|
|
|
# --------------------------------------------------
|
|
#region - Import Session Functions
|
|
# --------------------------------------------------
|
|
# Import functions from the current session into the RunspacePool sessionstate
|
|
|
|
if($ImportSessionFunctions)
|
|
{
|
|
# Import all session functions into the runspace session state from the current one
|
|
Get-ChildItem -Path Function:\ |
|
|
Where-Object -FilterScript {
|
|
$_.name -notlike '*:*'
|
|
} |
|
|
Select-Object -Property name -ExpandProperty name |
|
|
ForEach-Object -Process {
|
|
# Get the function code
|
|
$Definition = Get-Content -Path "function:\$_" -ErrorAction Stop
|
|
|
|
# Create a sessionstate function with the same name and code
|
|
$SessionStateFunction = New-Object -TypeName System.Management.Automation.Runspaces.SessionStateFunctionEntry -ArgumentList "$_", $Definition
|
|
|
|
# Add the function to the session state
|
|
$sessionstate.Commands.Add($SessionStateFunction)
|
|
}
|
|
}
|
|
#endregion
|
|
|
|
#Create runspace pool
|
|
$runspacepool = [runspacefactory]::CreateRunspacePool(1, $Throttle, $sessionstate, $Host)
|
|
$runspacepool.Open()
|
|
|
|
#Write-Verbose "Creating empty collection to hold runspace jobs"
|
|
$Script:runspaces = New-Object -TypeName System.Collections.ArrayList
|
|
|
|
#If inputObject is bound get a total count and set bound to true
|
|
$bound = $PSBoundParameters.keys -contains 'InputObject'
|
|
if(-not $bound)
|
|
{
|
|
[System.Collections.ArrayList]$allObjects = @()
|
|
}
|
|
|
|
<#
|
|
#write initial log entry
|
|
$log = "" | Select Date, Action, Runtime, Status, Details
|
|
$log.Date = Get-Date
|
|
$log.Action = "Batch processing started"
|
|
$log.Runtime = $null
|
|
$log.Status = "Started"
|
|
$log.Details = $null
|
|
#>
|
|
$timedOutTasks = $false
|
|
|
|
#endregion INIT
|
|
}
|
|
|
|
Process {
|
|
|
|
#add piped objects to all objects or set all objects to bound input object parameter
|
|
if($bound)
|
|
{
|
|
$allObjects = $InputObject
|
|
}
|
|
Else
|
|
{
|
|
[void]$allObjects.add( $InputObject )
|
|
}
|
|
}
|
|
|
|
End {
|
|
|
|
#Use Try/Finally to catch Ctrl+C and clean up.
|
|
Try
|
|
{
|
|
#counts for progress
|
|
$totalCount = $allObjects.count
|
|
$script:completedCount = 0
|
|
$startedCount = 0
|
|
|
|
foreach($object in $allObjects)
|
|
{
|
|
#region add scripts to runspace pool
|
|
|
|
#Create the powershell instance, set verbose if needed, supply the scriptblock and parameters
|
|
$powershell = [powershell]::Create()
|
|
|
|
if ($VerbosePreference -eq 'Continue')
|
|
{
|
|
[void]$powershell.AddScript({
|
|
$VerbosePreference = 'Continue'
|
|
})
|
|
}
|
|
|
|
[void]$powershell.AddScript($ScriptBlock).AddArgument($object)
|
|
|
|
if ($Parameter)
|
|
{
|
|
[void]$powershell.AddArgument($Parameter)
|
|
}
|
|
|
|
# $Using support from Boe Prox
|
|
if ($UsingVariableData)
|
|
{
|
|
Foreach($UsingVariable in $UsingVariableData)
|
|
{
|
|
#Write-Verbose "Adding $($UsingVariable.Name) with value: $($UsingVariable.Value)"
|
|
[void]$powershell.AddArgument($UsingVariable.Value)
|
|
}
|
|
}
|
|
|
|
#Add the runspace into the powershell instance
|
|
$powershell.RunspacePool = $runspacepool
|
|
|
|
#Create a temporary collection for each runspace
|
|
$temp = '' | Select-Object -Property PowerShell, StartTime, object, Runspace
|
|
$temp.PowerShell = $powershell
|
|
$temp.StartTime = Get-Date
|
|
$temp.object = $object
|
|
|
|
#Save the handle output when calling BeginInvoke() that will be used later to end the runspace
|
|
$temp.Runspace = $powershell.BeginInvoke()
|
|
$startedCount++
|
|
|
|
#Add the temp tracking info to $runspaces collection
|
|
#Write-Verbose ( "Adding {0} to collection at {1}" -f $temp.object, $temp.starttime.tostring() )
|
|
$null = $runspaces.Add($temp)
|
|
|
|
#loop through existing runspaces one time
|
|
Get-RunspaceData
|
|
|
|
#If we have more running than max queue (used to control timeout accuracy)
|
|
#Script scope resolves odd PowerShell 2 issue
|
|
$firstRun = $true
|
|
while ($runspaces.count -ge $script:MaxQueue)
|
|
{
|
|
#give verbose output
|
|
if($firstRun)
|
|
{
|
|
#Write-Verbose "$($runspaces.count) items running - exceeded $Script:MaxQueue limit."
|
|
}
|
|
$firstRun = $false
|
|
|
|
#run get-runspace data and sleep for a short while
|
|
Get-RunspaceData
|
|
Start-Sleep -Milliseconds $SleepTimer
|
|
}
|
|
|
|
#endregion add scripts to runspace pool
|
|
}
|
|
|
|
#Write-Verbose ( "Finish processing the remaining runspace jobs: {0}" -f ( @($runspaces | Where {$_.Runspace -ne $Null}).Count) )
|
|
Get-RunspaceData -wait
|
|
|
|
if (-not $Quiet)
|
|
{
|
|
Write-Progress -Activity 'Running Query' -Status 'Starting threads' -Completed
|
|
}
|
|
}
|
|
Finally
|
|
{
|
|
#Close the runspace pool, unless we specified no close on timeout and something timed out
|
|
if ( ($timedOutTasks -eq $false) -or ( ($timedOutTasks -eq $true) -and ($NoCloseOnTimeout -eq $false) ) )
|
|
{
|
|
If (-not($SuppressVerbose)){
|
|
Write-Verbose -Message 'Closing the runspace pool'
|
|
}
|
|
$runspacepool.close()
|
|
}
|
|
|
|
#collect garbage
|
|
[gc]::Collect()
|
|
}
|
|
}
|
|
}
|
|
|
|
Function Invoke-Ping
|
|
{
|
|
<#
|
|
.SYNOPSIS
|
|
Ping or test connectivity to systems in parallel
|
|
|
|
.DESCRIPTION
|
|
Ping or test connectivity to systems in parallel
|
|
|
|
Default action will run a ping against systems
|
|
If Quiet parameter is specified, we return an array of systems that responded
|
|
If Detail parameter is specified, we test WSMan, RemoteReg, RPC, RDP and/or SMB
|
|
|
|
.PARAMETER ComputerName
|
|
One or more computers to test
|
|
|
|
.PARAMETER Quiet
|
|
If specified, only return addresses that responded to Test-Connection
|
|
|
|
.PARAMETER Detail
|
|
Include one or more additional tests as specified:
|
|
WSMan via Test-WSMan
|
|
RemoteReg via Microsoft.Win32.RegistryKey
|
|
RPC via WMI
|
|
RDP via port 3389
|
|
SMB via \\ComputerName\C$
|
|
* All tests
|
|
|
|
.PARAMETER Timeout
|
|
Time in seconds before we attempt to dispose an individual query. Default is 20
|
|
|
|
.PARAMETER Throttle
|
|
Throttle query to this many parallel runspaces. Default is 100.
|
|
|
|
.PARAMETER NoCloseOnTimeout
|
|
Do not dispose of timed out tasks or attempt to close the runspace if threads have timed out
|
|
|
|
This will prevent the script from hanging in certain situations where threads become non-responsive, at the expense of leaking memory within the PowerShell host.
|
|
|
|
.EXAMPLE
|
|
Invoke-Ping Server1, Server2, Server3 -Detail *
|
|
|
|
# Check for WSMan, Remote Registry, Remote RPC, RDP, and SMB (via C$) connectivity against 3 machines
|
|
|
|
.EXAMPLE
|
|
$Computers | Invoke-Ping
|
|
|
|
# Ping computers in $Computers in parallel
|
|
|
|
.EXAMPLE
|
|
$Responding = $Computers | Invoke-Ping -Quiet
|
|
|
|
# Create a list of computers that successfully responded to Test-Connection
|
|
|
|
.LINK
|
|
https://gallery.technet.microsoft.com/scriptcenter/Invoke-Ping-Test-in-b553242a
|
|
|
|
.FUNCTIONALITY
|
|
Computers
|
|
|
|
#>
|
|
[cmdletbinding(DefaultParameterSetName='Ping')]
|
|
param(
|
|
[Parameter( ValueFromPipeline=$true,
|
|
ValueFromPipelineByPropertyName=$true,
|
|
Position=0)]
|
|
[string[]]$ComputerName,
|
|
|
|
[Parameter( ParameterSetName='Detail')]
|
|
[validateset("*","WSMan","RemoteReg","RPC","RDP","SMB")]
|
|
[string[]]$Detail,
|
|
|
|
[Parameter(ParameterSetName='Ping')]
|
|
[switch]$Quiet,
|
|
|
|
[int]$Timeout = 20,
|
|
|
|
[int]$Throttle = 100,
|
|
|
|
[switch]$NoCloseOnTimeout
|
|
)
|
|
Begin
|
|
{
|
|
|
|
#http://gallery.technet.microsoft.com/Run-Parallel-Parallel-377fd430
|
|
function Invoke-Parallel {
|
|
[cmdletbinding(DefaultParameterSetName='ScriptBlock')]
|
|
Param (
|
|
[Parameter(Mandatory=$false,position=0,ParameterSetName='ScriptBlock')]
|
|
[System.Management.Automation.ScriptBlock]$ScriptBlock,
|
|
|
|
[Parameter(Mandatory=$false,ParameterSetName='ScriptFile')]
|
|
[ValidateScript({test-path $_ -pathtype leaf})]
|
|
$ScriptFile,
|
|
|
|
[Parameter(Mandatory=$true,ValueFromPipeline=$true)]
|
|
[Alias('CN','__Server','IPAddress','Server','ComputerName')]
|
|
[PSObject]$InputObject,
|
|
|
|
[PSObject]$Parameter,
|
|
|
|
[switch]$ImportVariables,
|
|
|
|
[switch]$ImportModules,
|
|
|
|
[int]$Throttle = 20,
|
|
|
|
[int]$SleepTimer = 200,
|
|
|
|
[int]$RunspaceTimeout = 0,
|
|
|
|
[switch]$NoCloseOnTimeout = $false,
|
|
|
|
[int]$MaxQueue,
|
|
|
|
[validatescript({Test-Path (Split-Path $_ -parent)})]
|
|
[string]$LogFile = "C:\temp\log.log",
|
|
|
|
[switch] $Quiet = $false
|
|
)
|
|
|
|
Begin {
|
|
|
|
#No max queue specified? Estimate one.
|
|
#We use the script scope to resolve an odd PowerShell 2 issue where MaxQueue isn't seen later in the function
|
|
if( -not $PSBoundParameters.ContainsKey('MaxQueue') )
|
|
{
|
|
if($RunspaceTimeout -ne 0){ $script:MaxQueue = $Throttle }
|
|
else{ $script:MaxQueue = $Throttle * 3 }
|
|
}
|
|
else
|
|
{
|
|
$script:MaxQueue = $MaxQueue
|
|
}
|
|
|
|
#If they want to import variables or modules, create a clean runspace, get loaded items, use those to exclude items
|
|
if ($ImportVariables -or $ImportModules)
|
|
{
|
|
$StandardUserEnv = [powershell]::Create().addscript({
|
|
|
|
#Get modules and snapins in this clean runspace
|
|
$Modules = Get-Module | Select -ExpandProperty Name
|
|
$Snapins = Get-PSSnapin | Select -ExpandProperty Name
|
|
|
|
#Get variables in this clean runspace
|
|
#Called last to get vars like $? into session
|
|
$Variables = Get-Variable | Select -ExpandProperty Name
|
|
|
|
#Return a hashtable where we can access each.
|
|
@{
|
|
Variables = $Variables
|
|
Modules = $Modules
|
|
Snapins = $Snapins
|
|
}
|
|
}).invoke()[0]
|
|
|
|
if ($ImportVariables) {
|
|
#Exclude common parameters, bound parameters, and automatic variables
|
|
Function _temp {[cmdletbinding()] param() }
|
|
$VariablesToExclude = @( (Get-Command _temp | Select -ExpandProperty parameters).Keys + $PSBoundParameters.Keys + $StandardUserEnv.Variables )
|
|
Write-Verbose "Excluding variables $( ($VariablesToExclude | sort ) -join ", ")"
|
|
|
|
# we don't use 'Get-Variable -Exclude', because it uses regexps.
|
|
# One of the veriables that we pass is '$?'.
|
|
# There could be other variables with such problems.
|
|
# Scope 2 required if we move to a real module
|
|
$UserVariables = @( Get-Variable | Where { -not ($VariablesToExclude -contains $_.Name) } )
|
|
Write-Verbose "Found variables to import: $( ($UserVariables | Select -expandproperty Name | Sort ) -join ", " | Out-String).`n"
|
|
|
|
}
|
|
|
|
if ($ImportModules)
|
|
{
|
|
$UserModules = @( Get-Module | Where {$StandardUserEnv.Modules -notcontains $_.Name -and (Test-Path $_.Path -ErrorAction SilentlyContinue)} | Select -ExpandProperty Path )
|
|
$UserSnapins = @( Get-PSSnapin | Select -ExpandProperty Name | Where {$StandardUserEnv.Snapins -notcontains $_ } )
|
|
}
|
|
}
|
|
|
|
#region functions
|
|
|
|
Function Get-RunspaceData {
|
|
[cmdletbinding()]
|
|
param( [switch]$Wait )
|
|
|
|
#loop through runspaces
|
|
#if $wait is specified, keep looping until all complete
|
|
Do {
|
|
|
|
#set more to false for tracking completion
|
|
$more = $false
|
|
|
|
#Progress bar if we have inputobject count (bound parameter)
|
|
if (-not $Quiet) {
|
|
Write-Progress -Activity "Running Query" -Status "Starting threads"`
|
|
-CurrentOperation "$startedCount threads defined - $totalCount input objects - $script:completedCount input objects processed"`
|
|
-PercentComplete $( Try { $script:completedCount / $totalCount * 100 } Catch {0} )
|
|
}
|
|
|
|
#run through each runspace.
|
|
Foreach($runspace in $runspaces) {
|
|
|
|
#get the duration - inaccurate
|
|
$currentdate = Get-Date
|
|
$runtime = $currentdate - $runspace.startTime
|
|
$runMin = [math]::Round( $runtime.totalminutes ,2 )
|
|
|
|
#set up log object
|
|
$log = "" | select Date, Action, Runtime, Status, Details
|
|
$log.Action = "Removing:'$($runspace.object)'"
|
|
$log.Date = $currentdate
|
|
$log.Runtime = "$runMin minutes"
|
|
|
|
#If runspace completed, end invoke, dispose, recycle, counter++
|
|
If ($runspace.Runspace.isCompleted) {
|
|
|
|
$script:completedCount++
|
|
|
|
#check if there were errors
|
|
if($runspace.powershell.Streams.Error.Count -gt 0) {
|
|
|
|
#set the logging info and move the file to completed
|
|
$log.status = "CompletedWithErrors"
|
|
Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
foreach($ErrorRecord in $runspace.powershell.Streams.Error) {
|
|
Write-Error -ErrorRecord $ErrorRecord
|
|
}
|
|
}
|
|
else {
|
|
|
|
#add logging details and cleanup
|
|
$log.status = "Completed"
|
|
Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
}
|
|
|
|
#everything is logged, clean up the runspace
|
|
$runspace.powershell.EndInvoke($runspace.Runspace)
|
|
$runspace.powershell.dispose()
|
|
$runspace.Runspace = $null
|
|
$runspace.powershell = $null
|
|
|
|
}
|
|
|
|
#If runtime exceeds max, dispose the runspace
|
|
ElseIf ( $runspaceTimeout -ne 0 -and $runtime.totalseconds -gt $runspaceTimeout) {
|
|
|
|
$script:completedCount++
|
|
$timedOutTasks = $true
|
|
|
|
#add logging details and cleanup
|
|
$log.status = "TimedOut"
|
|
Write-Verbose ($log | ConvertTo-Csv -Delimiter ";" -NoTypeInformation)[1]
|
|
Write-Error "Runspace timed out at $($runtime.totalseconds) seconds for the object:`n$($runspace.object | out-string)"
|
|
|
|
#Depending on how it hangs, we could still get stuck here as dispose calls a synchronous method on the powershell instance
|
|
if (!$noCloseOnTimeout) { $runspace.powershell.dispose() }
|
|
$runspace.Runspace = $null
|
|
$runspace.powershell = $null
|
|
$completedCount++
|
|
|
|
}
|
|
|
|
#If runspace isn't null set more to true
|
|
ElseIf ($runspace.Runspace -ne $null ) {
|
|
$log = $null
|
|
$more = $true
|
|
}
|
|
}
|
|
|
|
#Clean out unused runspace jobs
|
|
$temphash = $runspaces.clone()
|
|
$temphash | Where { $_.runspace -eq $Null } | ForEach {
|
|
$Runspaces.remove($_)
|
|
}
|
|
|
|
#sleep for a bit if we will loop again
|
|
if($PSBoundParameters['Wait']){ Start-Sleep -milliseconds $SleepTimer }
|
|
|
|
#Loop again only if -wait parameter and there are more runspaces to process
|
|
} while ($more -and $PSBoundParameters['Wait'])
|
|
|
|
#End of runspace function
|
|
}
|
|
|
|
#endregion functions
|
|
|
|
#region Init
|
|
|
|
if($PSCmdlet.ParameterSetName -eq 'ScriptFile')
|
|
{
|
|
$ScriptBlock = [scriptblock]::Create( $(Get-Content $ScriptFile | out-string) )
|
|
}
|
|
elseif($PSCmdlet.ParameterSetName -eq 'ScriptBlock')
|
|
{
|
|
#Start building parameter names for the param block
|
|
[string[]]$ParamsToAdd = '$_'
|
|
if( $PSBoundParameters.ContainsKey('Parameter') )
|
|
{
|
|
$ParamsToAdd += '$Parameter'
|
|
}
|
|
|
|
$UsingVariableData = $Null
|
|
|
|
|
|
# This code enables $Using support through the AST.
|
|
# This is entirely from Boe Prox, and his https://github.com/proxb/PoshRSJob module; all credit to Boe!
|
|
|
|
if($PSVersionTable.PSVersion.Major -gt 2)
|
|
{
|
|
#Extract using references
|
|
$UsingVariables = $ScriptBlock.ast.FindAll({$args[0] -is [System.Management.Automation.Language.UsingExpressionAst]},$True)
|
|
|
|
If ($UsingVariables)
|
|
{
|
|
$List = New-Object 'System.Collections.Generic.List`1[System.Management.Automation.Language.VariableExpressionAst]'
|
|
ForEach ($Ast in $UsingVariables)
|
|
{
|
|
[void]$list.Add($Ast.SubExpression)
|
|
}
|
|
|
|
$UsingVar = $UsingVariables | Group Parent | ForEach {$_.Group | Select -First 1}
|
|
|
|
#Extract the name, value, and create replacements for each
|
|
$UsingVariableData = ForEach ($Var in $UsingVar) {
|
|
Try
|
|
{
|
|
$Value = Get-Variable -Name $Var.SubExpression.VariablePath.UserPath -ErrorAction Stop
|
|
$NewName = ('$__using_{0}' -f $Var.SubExpression.VariablePath.UserPath)
|
|
[pscustomobject]@{
|
|
Name = $Var.SubExpression.Extent.Text
|
|
Value = $Value.Value
|
|
NewName = $NewName
|
|
NewVarName = ('__using_{0}' -f $Var.SubExpression.VariablePath.UserPath)
|
|
}
|
|
$ParamsToAdd += $NewName
|
|
}
|
|
Catch
|
|
{
|
|
Write-Error "$($Var.SubExpression.Extent.Text) is not a valid Using: variable!"
|
|
}
|
|
}
|
|
|
|
$NewParams = $UsingVariableData.NewName -join ', '
|
|
$Tuple = [Tuple]::Create($list, $NewParams)
|
|
$bindingFlags = [Reflection.BindingFlags]"Default,NonPublic,Instance"
|
|
$GetWithInputHandlingForInvokeCommandImpl = ($ScriptBlock.ast.gettype().GetMethod('GetWithInputHandlingForInvokeCommandImpl',$bindingFlags))
|
|
|
|
$StringScriptBlock = $GetWithInputHandlingForInvokeCommandImpl.Invoke($ScriptBlock.ast,@($Tuple))
|
|
|
|
$ScriptBlock = [scriptblock]::Create($StringScriptBlock)
|
|
|
|
Write-Verbose $StringScriptBlock
|
|
}
|
|
}
|
|
|
|
$ScriptBlock = $ExecutionContext.InvokeCommand.NewScriptBlock("param($($ParamsToAdd -Join ", "))`r`n" + $Scriptblock.ToString())
|
|
}
|
|
else
|
|
{
|
|
Throw "Must provide ScriptBlock or ScriptFile"; Break
|
|
}
|
|
|
|
Write-Debug "`$ScriptBlock: $($ScriptBlock | Out-String)"
|
|
Write-Verbose "Creating runspace pool and session states"
|
|
|
|
#If specified, add variables and modules/snapins to session state
|
|
$sessionstate = [System.Management.Automation.Runspaces.InitialSessionState]::CreateDefault()
|
|
if ($ImportVariables)
|
|
{
|
|
if($UserVariables.count -gt 0)
|
|
{
|
|
foreach($Variable in $UserVariables)
|
|
{
|
|
$sessionstate.Variables.Add( (New-Object -TypeName System.Management.Automation.Runspaces.SessionStateVariableEntry -ArgumentList $Variable.Name, $Variable.Value, $null) )
|
|
}
|
|
}
|
|
}
|
|
if ($ImportModules)
|
|
{
|
|
if($UserModules.count -gt 0)
|
|
{
|
|
foreach($ModulePath in $UserModules)
|
|
{
|
|
$sessionstate.ImportPSModule($ModulePath)
|
|
}
|
|
}
|
|
if($UserSnapins.count -gt 0)
|
|
{
|
|
foreach($PSSnapin in $UserSnapins)
|
|
{
|
|
[void]$sessionstate.ImportPSSnapIn($PSSnapin, [ref]$null)
|
|
}
|
|
}
|
|
}
|
|
|
|
#Create runspace pool
|
|
$runspacepool = [runspacefactory]::CreateRunspacePool(1, $Throttle, $sessionstate, $Host)
|
|
$runspacepool.Open()
|
|
|
|
Write-Verbose "Creating empty collection to hold runspace jobs"
|
|
$Script:runspaces = New-Object System.Collections.ArrayList
|
|
|
|
#If inputObject is bound get a total count and set bound to true
|
|
$global:__bound = $false
|
|
$allObjects = @()
|
|
if( $PSBoundParameters.ContainsKey("inputObject") ){
|
|
$global:__bound = $true
|
|
}
|
|
|
|
#write initial log entry
|
|
$log = "" | Select Date, Action, Runtime, Status, Details
|
|
$log.Date = Get-Date
|
|
$log.Action = "Batch processing started"
|
|
$log.Runtime = $null
|
|
$log.Status = "Started"
|
|
$log.Details = $null
|
|
|
|
$timedOutTasks = $false
|
|
|
|
#endregion INIT
|
|
}
|
|
|
|
Process {
|
|
|
|
#add piped objects to all objects or set all objects to bound input object parameter
|
|
if( -not $global:__bound ){
|
|
$allObjects += $inputObject
|
|
}
|
|
else{
|
|
$allObjects = $InputObject
|
|
}
|
|
}
|
|
|
|
End {
|
|
|
|
#Use Try/Finally to catch Ctrl+C and clean up.
|
|
Try
|
|
{
|
|
#counts for progress
|
|
$totalCount = $allObjects.count
|
|
$script:completedCount = 0
|
|
$startedCount = 0
|
|
|
|
foreach($object in $allObjects){
|
|
|
|
#region add scripts to runspace pool
|
|
|
|
#Create the powershell instance, set verbose if needed, supply the scriptblock and parameters
|
|
$powershell = [powershell]::Create()
|
|
|
|
if ($VerbosePreference -eq 'Continue')
|
|
{
|
|
[void]$PowerShell.AddScript({$VerbosePreference = 'Continue'})
|
|
}
|
|
|
|
[void]$PowerShell.AddScript($ScriptBlock).AddArgument($object)
|
|
|
|
if ($parameter)
|
|
{
|
|
[void]$PowerShell.AddArgument($parameter)
|
|
}
|
|
|
|
# $Using support from Boe Prox
|
|
if ($UsingVariableData)
|
|
{
|
|
Foreach($UsingVariable in $UsingVariableData) {
|
|
Write-Verbose "Adding $($UsingVariable.Name) with value: $($UsingVariable.Value)"
|
|
[void]$PowerShell.AddArgument($UsingVariable.Value)
|
|
}
|
|
}
|
|
|
|
#Add the runspace into the powershell instance
|
|
$powershell.RunspacePool = $runspacepool
|
|
|
|
#Create a temporary collection for each runspace
|
|
$temp = "" | Select-Object PowerShell, StartTime, object, Runspace
|
|
$temp.PowerShell = $powershell
|
|
$temp.StartTime = Get-Date
|
|
$temp.object = $object
|
|
|
|
#Save the handle output when calling BeginInvoke() that will be used later to end the runspace
|
|
$temp.Runspace = $powershell.BeginInvoke()
|
|
$startedCount++
|
|
|
|
#Add the temp tracking info to $runspaces collection
|
|
Write-Verbose ( "Adding {0} to collection at {1}" -f $temp.object, $temp.starttime.tostring() )
|
|
$runspaces.Add($temp) | Out-Null
|
|
|
|
#loop through existing runspaces one time
|
|
if($ShowRunpaceErrors){
|
|
Get-RunspaceData
|
|
}else{
|
|
Get-RunspaceData -ErrorAction SilentlyContinue
|
|
}
|
|
|
|
#If we have more running than max queue (used to control timeout accuracy)
|
|
#Script scope resolves odd PowerShell 2 issue
|
|
$firstRun = $true
|
|
while ($runspaces.count -ge $Script:MaxQueue) {
|
|
|
|
#give verbose output
|
|
if($firstRun){
|
|
Write-Verbose "$($runspaces.count) items running - exceeded $Script:MaxQueue limit."
|
|
}
|
|
$firstRun = $false
|
|
|
|
#run get-runspace data and sleep for a short while
|
|
Get-RunspaceData
|
|
Start-Sleep -Milliseconds $sleepTimer
|
|
|
|
}
|
|
|
|
#endregion add scripts to runspace pool
|
|
}
|
|
|
|
Write-Verbose ( "Finish processing the remaining runspace jobs: {0}" -f ( @($runspaces | Where {$_.Runspace -ne $Null}).Count) )
|
|
Get-RunspaceData -wait
|
|
|
|
if (-not $quiet) {
|
|
Write-Progress -Activity "Running Query" -Status "Starting threads" -Completed
|
|
}
|
|
|
|
}
|
|
Finally
|
|
{
|
|
#Close the runspace pool, unless we specified no close on timeout and something timed out
|
|
if ( ($timedOutTasks -eq $false) -or ( ($timedOutTasks -eq $true) -and ($noCloseOnTimeout -eq $false) ) ) {
|
|
Write-Verbose "Closing the runspace pool"
|
|
$runspacepool.close()
|
|
}
|
|
|
|
#collect garbage
|
|
[gc]::Collect()
|
|
}
|
|
}
|
|
}
|
|
|
|
Write-Verbose "PSBoundParameters = $($PSBoundParameters | Out-String)"
|
|
|
|
$bound = $PSBoundParameters.keys -contains "ComputerName"
|
|
if(-not $bound)
|
|
{
|
|
[System.Collections.ArrayList]$AllComputers = @()
|
|
}
|
|
}
|
|
Process
|
|
{
|
|
|
|
#Handle both pipeline and bound parameter. We don't want to stream objects, defeats purpose of parallelizing work
|
|
if($bound)
|
|
{
|
|
$AllComputers = $ComputerName
|
|
}
|
|
Else
|
|
{
|
|
foreach($Computer in $ComputerName)
|
|
{
|
|
$AllComputers.add($Computer) | Out-Null
|
|
}
|
|
}
|
|
|
|
}
|
|
End
|
|
{
|
|
|
|
#Built up the parameters and run everything in parallel
|
|
$params = @($Detail, $Quiet)
|
|
$splat = @{
|
|
Throttle = $Throttle
|
|
RunspaceTimeout = $Timeout
|
|
InputObject = $AllComputers
|
|
parameter = $params
|
|
}
|
|
if($NoCloseOnTimeout)
|
|
{
|
|
$splat.add('NoCloseOnTimeout',$True)
|
|
}
|
|
|
|
Invoke-Parallel @splat -ScriptBlock {
|
|
|
|
$computer = $_.trim()
|
|
$detail = $parameter[0]
|
|
$quiet = $parameter[1]
|
|
|
|
#They want detail, define and run test-server
|
|
if($detail)
|
|
{
|
|
Try
|
|
{
|
|
#Modification of jrich's Test-Server function: https://gallery.technet.microsoft.com/scriptcenter/Powershell-Test-Server-e0cdea9a
|
|
Function Test-Server{
|
|
[cmdletBinding()]
|
|
param(
|
|
[parameter(
|
|
Mandatory=$true,
|
|
ValueFromPipeline=$true)]
|
|
[string[]]$ComputerName,
|
|
[switch]$All,
|
|
[parameter(Mandatory=$false)]
|
|
[switch]$CredSSP,
|
|
[switch]$RemoteReg,
|
|
[switch]$RDP,
|
|
[switch]$RPC,
|
|
[switch]$SMB,
|
|
[switch]$WSMAN,
|
|
[switch]$IPV6,
|
|
[Management.Automation.PSCredential]$Credential
|
|
)
|
|
begin
|
|
{
|
|
$total = Get-Date
|
|
$results = @()
|
|
if($credssp -and -not $Credential)
|
|
{
|
|
Throw "Must supply Credentials with CredSSP test"
|
|
}
|
|
|
|
[string[]]$props = write-output Name, IP, Domain, Ping, WSMAN, CredSSP, RemoteReg, RPC, RDP, SMB
|
|
|
|
#Hash table to create PSObjects later, compatible with ps2...
|
|
$Hash = @{}
|
|
foreach($prop in $props)
|
|
{
|
|
$Hash.Add($prop,$null)
|
|
}
|
|
|
|
function Test-Port{
|
|
[cmdletbinding()]
|
|
Param(
|
|
[string]$srv,
|
|
$port=135,
|
|
$timeout=3000
|
|
)
|
|
$ErrorActionPreference = "SilentlyContinue"
|
|
$tcpclient = new-Object system.Net.Sockets.TcpClient
|
|
$iar = $tcpclient.BeginConnect($srv,$port,$null,$null)
|
|
$wait = $iar.AsyncWaitHandle.WaitOne($timeout,$false)
|
|
if(-not $wait)
|
|
{
|
|
$tcpclient.Close()
|
|
Write-Verbose "Connection Timeout to $srv`:$port"
|
|
$false
|
|
}
|
|
else
|
|
{
|
|
Try
|
|
{
|
|
$tcpclient.EndConnect($iar) | out-Null
|
|
$true
|
|
}
|
|
Catch
|
|
{
|
|
write-verbose "Error for $srv`:$port`: $_"
|
|
$false
|
|
}
|
|
$tcpclient.Close()
|
|
}
|
|
}
|
|
}
|
|
|
|
process
|
|
{
|
|
foreach($name in $computername)
|
|
{
|
|
$dt = $cdt= Get-Date
|
|
Write-verbose "Testing: $Name"
|
|
$failed = 0
|
|
try{
|
|
$DNSEntity = [Net.Dns]::GetHostEntry($name)
|
|
$domain = ($DNSEntity.hostname).replace("$name.","")
|
|
$ips = $DNSEntity.AddressList | %{
|
|
if(-not ( -not $IPV6 -and $_.AddressFamily -like "InterNetworkV6" ))
|
|
{
|
|
$_.IPAddressToString
|
|
}
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
$rst = New-Object -TypeName PSObject -Property $Hash | Select -Property $props
|
|
$rst.name = $name
|
|
$results += $rst
|
|
$failed = 1
|
|
}
|
|
Write-verbose "DNS: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
if($failed -eq 0){
|
|
foreach($ip in $ips)
|
|
{
|
|
|
|
$rst = New-Object -TypeName PSObject -Property $Hash | Select -Property $props
|
|
$rst.name = $name
|
|
$rst.ip = $ip
|
|
$rst.domain = $domain
|
|
|
|
if($RDP -or $All)
|
|
{
|
|
####RDP Check (firewall may block rest so do before ping
|
|
try{
|
|
$socket = New-Object Net.Sockets.TcpClient($name, 3389) -ErrorAction stop
|
|
if($socket -eq $null)
|
|
{
|
|
$rst.RDP = $false
|
|
}
|
|
else
|
|
{
|
|
$rst.RDP = $true
|
|
$socket.close()
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
$rst.RDP = $false
|
|
Write-Verbose "Error testing RDP: $_"
|
|
}
|
|
}
|
|
Write-verbose "RDP: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
#########ping
|
|
if(test-connection $ip -count 2 -Quiet)
|
|
{
|
|
Write-verbose "PING: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
$rst.ping = $true
|
|
|
|
if($WSMAN -or $All)
|
|
{
|
|
try{############wsman
|
|
Test-WSMan $ip -ErrorAction stop | Out-Null
|
|
$rst.WSMAN = $true
|
|
}
|
|
catch
|
|
{
|
|
$rst.WSMAN = $false
|
|
Write-Verbose "Error testing WSMAN: $_"
|
|
}
|
|
Write-verbose "WSMAN: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
if($rst.WSMAN -and $credssp) ########### credssp
|
|
{
|
|
try{
|
|
Test-WSMan $ip -Authentication Credssp -Credential $cred -ErrorAction stop
|
|
$rst.CredSSP = $true
|
|
}
|
|
catch
|
|
{
|
|
$rst.CredSSP = $false
|
|
Write-Verbose "Error testing CredSSP: $_"
|
|
}
|
|
Write-verbose "CredSSP: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
}
|
|
}
|
|
if($RemoteReg -or $All)
|
|
{
|
|
try ########remote reg
|
|
{
|
|
[Microsoft.Win32.RegistryKey]::OpenRemoteBaseKey([Microsoft.Win32.RegistryHive]::LocalMachine, $ip) | Out-Null
|
|
$rst.remotereg = $true
|
|
}
|
|
catch
|
|
{
|
|
$rst.remotereg = $false
|
|
Write-Verbose "Error testing RemoteRegistry: $_"
|
|
}
|
|
Write-verbose "remote reg: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
}
|
|
if($RPC -or $All)
|
|
{
|
|
try ######### wmi
|
|
{
|
|
$w = [wmi] ''
|
|
$w.psbase.options.timeout = 15000000
|
|
$w.path = "\\$Name\root\cimv2:Win32_ComputerSystem.Name='$Name'"
|
|
$w | select none | Out-Null
|
|
$rst.RPC = $true
|
|
}
|
|
catch
|
|
{
|
|
$rst.rpc = $false
|
|
Write-Verbose "Error testing WMI/RPC: $_"
|
|
}
|
|
Write-verbose "WMI/RPC: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
}
|
|
if($SMB -or $All)
|
|
{
|
|
|
|
#Use set location and resulting errors. push and pop current location
|
|
try ######### C$
|
|
{
|
|
$path = "\\$name\c$"
|
|
Push-Location -Path $path -ErrorAction stop
|
|
$rst.SMB = $true
|
|
Pop-Location
|
|
}
|
|
catch
|
|
{
|
|
$rst.SMB = $false
|
|
Write-Verbose "Error testing SMB: $_"
|
|
}
|
|
Write-verbose "SMB: $((New-TimeSpan $dt ($dt = get-date)).totalseconds)"
|
|
|
|
}
|
|
}
|
|
else
|
|
{
|
|
$rst.ping = $false
|
|
$rst.wsman = $false
|
|
$rst.credssp = $false
|
|
$rst.remotereg = $false
|
|
$rst.rpc = $false
|
|
$rst.smb = $false
|
|
}
|
|
$results += $rst
|
|
}
|
|
}
|
|
Write-Verbose "Time for $($Name): $((New-TimeSpan $cdt ($dt)).totalseconds)"
|
|
Write-Verbose "----------------------------"
|
|
}
|
|
}
|
|
end
|
|
{
|
|
Write-Verbose "Time for all: $((New-TimeSpan $total ($dt)).totalseconds)"
|
|
Write-Verbose "----------------------------"
|
|
return $results
|
|
}
|
|
}
|
|
|
|
#Build up parameters for Test-Server and run it
|
|
$TestServerParams = @{
|
|
ComputerName = $Computer
|
|
ErrorAction = "Stop"
|
|
}
|
|
|
|
if($detail -eq "*"){
|
|
$detail = "WSMan","RemoteReg","RPC","RDP","SMB"
|
|
}
|
|
|
|
$detail | Select -Unique | Foreach-Object { $TestServerParams.add($_,$True) }
|
|
Test-Server @TestServerParams | Select -Property $( "Name", "IP", "Domain", "Ping" + $detail )
|
|
}
|
|
Catch
|
|
{
|
|
Write-Warning "Error with Test-Server: $_"
|
|
}
|
|
}
|
|
#We just want ping output
|
|
else
|
|
{
|
|
Try
|
|
{
|
|
#Pick out a few properties, add a status label. If quiet output, just return the address
|
|
$result = $null
|
|
if( $result = @( Test-Connection -ComputerName $computer -Count 2 -erroraction Stop ) )
|
|
{
|
|
$Output = $result | Select -first 1 -Property Address,
|
|
IPV4Address,
|
|
IPV6Address,
|
|
ResponseTime,
|
|
@{ label = "STATUS"; expression = {"Responding"} }
|
|
|
|
if( $quiet )
|
|
{
|
|
$Output.address
|
|
}
|
|
else
|
|
{
|
|
$Output
|
|
}
|
|
}
|
|
}
|
|
Catch
|
|
{
|
|
if(-not $quiet)
|
|
{
|
|
#Ping failed. I'm likely making inappropriate assumptions here, let me know if this is the case : )
|
|
if($_ -match "No such host is known")
|
|
{
|
|
$status = "Unknown host"
|
|
}
|
|
elseif($_ -match "Error due to lack of resources")
|
|
{
|
|
$status = "No Response"
|
|
}
|
|
else
|
|
{
|
|
$status = "Error: $_"
|
|
}
|
|
|
|
"" | Select -Property @{ label = "Address"; expression = {$computer} },
|
|
IPV4Address,
|
|
IPV6Address,
|
|
ResponseTime,
|
|
@{ label = "STATUS"; expression = {$status} }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
# ---------------------------------------------------
|
|
# Function: checksubnet
|
|
# Author: Luben Kirov
|
|
# Reference: http://www.gi-architects.co.uk/2016/02/powershell-check-if-ip-or-subnet-matchesfits/
|
|
# ---------------------------------------------------
|
|
# The function will check ip to ip, ip to subnet, subnet to ip or subnet to subnet belong to each other and return true or false and the direction of the check
|
|
#////////////////////////////////////////////////////////////////////////
|
|
function checkSubnet ([string]$addr1, [string]$addr2)
|
|
{
|
|
# Separate the network address and lenght
|
|
$network1, [int]$subnetlen1 = $addr1.Split('/')
|
|
$network2, [int]$subnetlen2 = $addr2.Split('/')
|
|
|
|
|
|
#Convert network address to binary
|
|
[uint32] $unetwork1 = NetworkToBinary $network1
|
|
|
|
[uint32] $unetwork2 = NetworkToBinary $network2
|
|
|
|
|
|
#Check if subnet length exists and is less then 32(/32 is host, single ip so no calculation needed) if so convert to binary
|
|
if($subnetlen1 -lt 32){
|
|
[uint32] $mask1 = SubToBinary $subnetlen1
|
|
}
|
|
|
|
if($subnetlen2 -lt 32){
|
|
[uint32] $mask2 = SubToBinary $subnetlen2
|
|
}
|
|
|
|
#Compare the results
|
|
if($mask1 -and $mask2){
|
|
# If both inputs are subnets check which is smaller and check if it belongs in the larger one
|
|
if($mask1 -lt $mask2){
|
|
return CheckSubnetToNetwork $unetwork1 $mask1 $unetwork2
|
|
}else{
|
|
return CheckNetworkToSubnet $unetwork2 $mask2 $unetwork1
|
|
}
|
|
}ElseIf($mask1){
|
|
# If second input is address and first input is subnet check if it belongs
|
|
return CheckSubnetToNetwork $unetwork1 $mask1 $unetwork2
|
|
}ElseIf($mask2){
|
|
# If first input is address and second input is subnet check if it belongs
|
|
return CheckNetworkToSubnet $unetwork2 $mask2 $unetwork1
|
|
}Else{
|
|
# If both inputs are ip check if they match
|
|
CheckNetworkToNetwork $unetwork1 $unetwork2
|
|
}
|
|
}
|
|
|
|
function CheckNetworkToSubnet ([uint32]$un2, [uint32]$ma2, [uint32]$un1)
|
|
{
|
|
$ReturnArray = "" | Select-Object -Property Condition,Direction
|
|
|
|
if($un2 -eq ($ma2 -band $un1)){
|
|
$ReturnArray.Condition = $True
|
|
$ReturnArray.Direction = "Addr1ToAddr2"
|
|
return $ReturnArray
|
|
}else{
|
|
$ReturnArray.Condition = $False
|
|
$ReturnArray.Direction = "Addr1ToAddr2"
|
|
return $ReturnArray
|
|
}
|
|
}
|
|
|
|
function CheckSubnetToNetwork ([uint32]$un1, [uint32]$ma1, [uint32]$un2)
|
|
{
|
|
$ReturnArray = "" | Select-Object -Property Condition,Direction
|
|
|
|
if($un1 -eq ($ma1 -band $un2)){
|
|
$ReturnArray.Condition = $True
|
|
$ReturnArray.Direction = "Addr2ToAddr1"
|
|
return $ReturnArray
|
|
}else{
|
|
$ReturnArray.Condition = $False
|
|
$ReturnArray.Direction = "Addr2ToAddr1"
|
|
return $ReturnArray
|
|
}
|
|
}
|
|
|
|
function CheckNetworkToNetwork ([uint32]$un1, [uint32]$un2)
|
|
{
|
|
$ReturnArray = "" | Select-Object -Property Condition,Direction
|
|
|
|
if($un1 -eq $un2){
|
|
$ReturnArray.Condition = $True
|
|
$ReturnArray.Direction = "Addr1ToAddr2"
|
|
return $ReturnArray
|
|
}else{
|
|
$ReturnArray.Condition = $False
|
|
$ReturnArray.Direction = "Addr1ToAddr2"
|
|
return $ReturnArray
|
|
}
|
|
}
|
|
|
|
function SubToBinary ([int]$sub)
|
|
{
|
|
return ((-bnot [uint32]0) -shl (32 - $sub))
|
|
}
|
|
|
|
function NetworkToBinary ($network)
|
|
{
|
|
$a = [uint32[]]$network.split('.')
|
|
return ($a[0] -shl 24) + ($a[1] -shl 16) + ($a[2] -shl 8) + $a[3]
|
|
}
|