这是一种使用多线程从用户输入列表中返回上次登录的方法。对于 54 个 DC,我发现 6 个线程是最佳点。根据环境中的 DC 数量调整线程数。
######### Begin Functions ###########
Function Get-FilePath(){
<#
.SYNOPSIS
Prompts user to select folder for input/output file
.DESCRIPTION
Prompts user to select folder for input/output file using System.Windows.Forms.OpenFileDialog. This would be used to inport data from a csv/txt file and/or specifying the output file.
.PARAMETER initialDir
Defines the starting directory for the Open File Dialog.
.EXAMPLE
Open File Dialog box with initial directory sent to %AppData%
Get-Filepath -initialDir "%appdata%"
.EXAMPLE
Open File Dialog box with initial directory sent to C:\Scripts
Get-Filepath -initialDir "C:\Scripts"
.NOTES
If running from ISE, the Open File Dialog my open behind the ISE window. Simply move or minimize the ISE window.
#>
[cmdletbinding()]
Param(
[Parameter(Mandatory=$false)]
[String]$initialDir=[Environment]::GetFolderPath("Desktop")
)
Write-Verbose "Loading System.Windows.Forms"
[void][System.Reflection.Assembly]::LoadWithPartialName("System.Windows.Forms")
Write-Verbose "Initial Directory set to $initialDir"
$OpenFileDialog = New-Object System.Windows.Forms.OpenFileDialog
$OpenFileDialog.InitialDirectory = $initialDir
$OpenFileDialog.Filter = "CSV Files (*.csv)|*.csv|Text Files (*.txt)| *.txt|Input Files (*.txt;*.csv)|*.txt;*.csv|All Files (*.*)|*.*"
$OpenFileDialog.FilterIndex = 3
$OpenFileDialog.Title = "Select the input file"
$OpenFileDialog.ShowHelp = $true
$ShowDiag = $OpenFileDialog.ShowDialog()
If($ShowDiag -eq "Cancel"){
break
}
$OpenFileDialog.FileName
}
function Get-InputFile([string]$inputFilePath){
if($inputFilePath -is [String] -and $inputFilePath -eq [String]::Empty){
$inputFilePath = Get-FilePath -initialDir $PSScriptRoot
$inputFile = Get-Content $inputFilePath
}
else{
$inputFile = Get-Content $inputFilePath
}
return $inputFile,$inputFilePath
}
function WriteToFile($expFullPath, $result){
#calls streamwriter object to write to file quickly
$writer = New-Object System.IO.StreamWriter $expFullPath
#converts the output array to csv then writes each line to Export file csv.
$result | ConvertTo-CSV -NoTypeInformation | ForEach-Object{
If($_ -ne $null){
$writer.WriteLine( $_ )
}
}
# Closes Write function.
$writer.Close()
}
Function Invoke-Multi-Thread
{
Param($ScriptBlock = $MainSB,
$UserList = $UserList,
$DCList = $DCList,
$OutPath = $OutPath,
$MaxThreads = 5,
$SleepTimer = 30,
$MaxWaitAtEnd = 60,
$OutputType = "Text")
"Killing existing jobs . . ."
Get-Job | Remove-Job -Force
"Done."
$i = 0
ForEach ($User in $UserList){
While ($(Get-Job -state running).count -ge $MaxThreads){
Write-Progress -Activity "Creating Server List" `
-Status "Waiting for threads to close" `
-CurrentOperation "$i threads created - $($(Get-Job -state running).count) threads open" `
-PercentComplete ($i / $UserList.count * 100)
Start-Sleep -Milliseconds $SleepTimer
}
#"Starting job - $Input"
$i++
# Start-Job -FilePath $ScriptFile -ArgumentList $Computer -Name $Computer | Out-Null
Start-Job -ScriptBlock $ScriptBlock -ArgumentList $User,$DCList,$OutPath -Name $User | Out-Null
Write-Progress -Activity "Creating Server List" `
-Status "Starting Threads" `
-CurrentOperation "$i threads created - $($(Get-Job -state running).count) threads open" `
-PercentComplete ($i / $UserList.count * 100)
}
$Complete = Get-date
While ($(Get-Job -State Running).count -gt 0){
$UserListStillRunning = ""
ForEach ($System in $(Get-Job -state running)){$UserListStillRunning += ", $($System.name)"}
$UserListStillRunning = $UserListStillRunning.Substring(2)
Write-Progress -Activity "Creating Input List" `
-Status "$($(Get-Job -State Running).count) threads remaining" `
-CurrentOperation "$UserListStillRunning" `
-PercentComplete ($(Get-Job -State Completed).count / $(Get-Job).count * 100)
If ($(New-TimeSpan $Complete $(Get-Date)).totalseconds -ge $MaxWaitAtEnd){"Killing all jobs still running . . .";Get-Job -State Running | Remove-Job -Force}
Start-Sleep -Milliseconds $SleepTimer
}
#"Reading all jobs"
If ($OutputType -eq "Text"){
ForEach($Job in Get-Job){
# "$($Job.Name)"
# "****************************************"
Receive-Job $Job
}
}
ElseIf($OutputType -eq "GridView"){
Get-Job | Receive-Job | Select-Object * -ExcludeProperty RunspaceId | out-gridview
}
} # End Invoke-Multi-Thread
$MainSB = {
$user = $args[0].tostring()
$DCList = $args[1]
$OutPath = $args[2]
try {
$dn = Get-ADUser $user |select distinguishedName
$UserResults= @()
foreach ($server in $DCList)
{
$tempRes = @()
$tempRes = "" |select Server, Name, LastLogon
$res = Get-ADObject -Identity $dn.distinguishedName -Server $server -Properties lastLogon |select lastLogon
$tempRes.Server = $server
$tempRes.Name = $user
$tempRes.LastLogon = [datetime]::FromFileTime($res.lastLogon)
$UserResults += $tempRes
}
$recentTime="01/10/1500 1:00 AM"
$recentServer=""
foreach ($result in $UserResults)
{
if ($result.LastLogon -gt $recentTime)
{
$recentTime = $result.LastLogon
$recentServer = $result.Server
$recentUser = $result.Name
}}
$LastLogonRes = @()
$LastLogonRes = "" |select Server, Name, LastLogon
$LastLogonRes.Name = $recentUser
$LastLogonRes.LastLogon = $recentTime
$LastLogonRes.Server = $recentServer
return $LastLogonRes
}
catch
{
$LastLogonRes = @()
$LastLogonRes = "" |select Server, Name, LastLogon
$LastLogonRes.Name = $user
$LastLogonRes.LastLogon = "N/A"
$LastLogonRes.Server = "N/A"
return $LastLogonRes
}
}
###### End Main Script Block #############
$result = New-Object System.Collections.ArrayList
$sw = [System.Diagnostics.Stopwatch]::StartNew()
$sw2 = [System.Diagnostics.Stopwatch]::StartNew()
$interval = 0
$inputcount = $inputFile.count
$lastinterval = 0
# Start Date and time for file name
$StartDate = Get-Date
$runDate = $StartDate.ToString("yyyy.MM.dd.HHmmss")
#name of script
$scriptName = ($MyInvocation.MyCommand.Name).Split(".")[-2]
$scriptpath = $MyInvocation.MyCommand.Path
$scriptcontents = $MyInvocation.MyCommand.ScriptBlock
$scriptlastmodified = (Get-item -Path $scriptpath).LastWriteTime
#job names
$jobName = $runDate
$jobPath = $PSScriptRoot + "\jobs\" + $jobName + "\"
New-Item -Path $jobPath -ItemType directory | Out-Null
#error file name
$logFile = $runDate + "_" + $ScriptName + "_ErrorLog.txt"
$logFullPath = $jobPath + $logFile
Start-Transcript -Path $logFullPath
#lastmodified info for log
$hash = CertUtil -hashfile $MyInvocation.MyCommand.Name MD5
Write-Host @"
Script last modified: $scriptlastmodified
$($hash[1]) - is the MD5 CheckSum of the following Script:
$($MyInvocation.MyCommand.Path)$($MyInvocation.MyCommand)
"@
#lists any commandline parameters used
$boundparams = $MyInvocation.BoundParameters
if($($boundparams.count) -gt 0){
Write-Host "
There were $($boundparams.count) command line parameters used"
Write-Host "The parameters were:"$boundparams
}
Write-Host @"
"@
#name of export file
$expFileName = $runDate + "_" + $scriptName + "_Export.csv"
$expFullPath = $jobPath + $expFileName
#selects input file
#GetsData from input file
$inputFile,$inputFilePath = Get-InputFile($inputFilePath)
#Copies Input file to output folder.
Copy-Item "$inputFilePath" "$jobPath"
$dcs = Get-ADDomainController -Filter {Name -like "*"}
$DCList=@()
foreach($dc in $dcs)
{
$DCList+= $dc.HostName
}
$users= gc $inputFilePath
write-host "Users count: " $users.count
$stopwatch = [system.diagnostics.stopwatch]::StartNew()
$LastLogonResults = Invoke-Multi-Thread -UserList $users -DCList $DCList -ScriptBlock $MainSB -MaxThreads 6 -OutPath $Out_Folder
$LastLogonResults = $LastLogonResults |select Server, name, lastLogon
#writes the results to file
WriteToFile($expFullPath) ($LastLogonResults)
$stopwatch.stop()
write-host $stopwatch.Elapsed.TotalSeconds
#closes progress bar
$sw.Stop()
Write-Progress -Activity $ScriptName -Status "Exported Results" -Completed
#exports csv rowcount and hashfile
$hash2 = CertUtil -hashfile $expFullPath MD5
Write-Host @"
$($hash2[1]) - is the MD5 CheckSum of the following Outputfile:
$expFullPath
"@
Write-Host "Script Run Time: (Days.Hours:Mins:Secs:Milliseconds)"
$sw.Elapsed.ToString("dd\.hh\:mm\:ss\:fff")
#closes progress bar
Write-Progress -Activity $ScriptName -Status "Exported Results" -Completed
Write-Host "OutFolder:$jobPath"
Write-Host "OutLog:$logFullPath"
Write-Host "OutData:$expFullPath"
Stop-Transcript
Write-host "Script Complete"
Pause