Last active
March 7, 2016 15:30
-
-
Save watson0x90/f04aba16c41f028dcff5 to your computer and use it in GitHub Desktop.
Query Windows Security Event Log via PowerShell HTTP Server
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#requires -Version 2 | |
$header = @" | |
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Frameset//EN" "http://www.w3.org/TR/html4/frameset.dtd"> | |
<html><head><title>Windows Event Logs</title> | |
<meta charset="utf-8"> | |
<meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
<style type="text/css"> | |
<!- | |
/* Begin Tab Def */ | |
.tabs { | |
width: 100%; | |
overflow: auto | |
} | |
.tabs li.tab { | |
float: none; | |
display: inline; | |
} | |
.tabs input[type="radio"] { | |
display: none; | |
} | |
.tabs .tab>label { | |
padding: 8px 1em; | |
border-radius: 5px 5px 0 0; | |
border: 1px solid #ddd; | |
border-bottom: none; | |
cursor: pointer; | |
z-index: 3; | |
background-color: #eee; | |
} | |
.tabs .tab-content { | |
z-index: 2; | |
display: none; | |
float: left; | |
padding: 1em; | |
left: 0; | |
border: 1px solid #ddd; | |
margin-top: 10px; | |
width: 95%; | |
} | |
.tabs [id^="tab"]:checked + label { | |
background-color: #fff; | |
border-bottom: 3px solid #fff; | |
} | |
.tabs [id^="tab"]:checked ~ [id^="tab-content"] { | |
display: block; | |
} | |
/* End Tab Def */ | |
iframe { | |
border: 0; | |
width: 100%; | |
height: 100%; | |
} | |
body { | |
font-family: Verdana, Geneva, Arial, Helvetica, sans-serif; | |
} | |
#report { width: 835px; } | |
table{ | |
border-collapse: collapse; | |
border: none; | |
font: 10pt Verdana, Geneva, Arial, Helvetica, sans-serif; | |
color: black; | |
margin-bottom: 10px; | |
border: 1px solid black; | |
} | |
table td{ | |
font-size: 12px; | |
padding-left: 3px; | |
padding-right: 5px; | |
text-align: left; | |
border: 1px solid; | |
border-color: #3C3C3C; | |
} | |
table th { | |
font-size: 12px; | |
font-weight: bold; | |
padding-left: 0px; | |
padding-right: 20px; | |
text-align: left; | |
border: 1px solid; | |
border-color: #3C3C3C; | |
background-color: #4CAF50; | |
color: white; | |
} | |
h2{ clear: both; font-size: 130%;color:#354B5E; } | |
h3{ | |
clear: both; | |
font-size: 75%; | |
margin-left: 20px; | |
margin-top: 30px; | |
color:#475F77; | |
} | |
p{ margin-left: 20px; font-size: 12px; } | |
table.list{ float: left; } | |
table.list td:nth-child(1){ | |
font-weight: bold; | |
border-right: 1px grey solid; | |
text-align: right; | |
} | |
ul.menu { | |
list-style-type: none; | |
margin: 0; | |
padding: 0; | |
overflow: hidden; | |
background-color: #333; | |
} | |
li.menu { | |
float: left; | |
} | |
li.menu a.menu { | |
display: block; | |
color: white; | |
text-align: center; | |
padding: 14px 16px; | |
text-decoration: none; | |
} | |
li.menu a:hover:not(.active).menu { | |
background-color: #111; | |
} | |
#active { | |
background-color: #4CAF50; | |
} | |
#stop { | |
background-color: #f44336; | |
} | |
.warning { | |
border: 1px solid; | |
margin: 10px 0px; | |
padding:15px 10px 15px 50px; | |
background-repeat: no-repeat; | |
background-position: 10px center; | |
color: #9F6000; | |
background-size: 35px 35px; | |
background-color: #FEEFB3; | |
background-image: url(''); | |
} | |
.button { | |
background-color: #4CAF50; /* Green */ | |
border: none; | |
color: white; | |
padding: 10px 13px; | |
text-align: center; | |
text-decoration: none; | |
display: inline-block; | |
font-size: 16px; | |
margin: 4px 2px; | |
cursor: pointer; | |
} | |
.button1 {background-color: #4CAF50;} /* Green */ | |
.button1 {border-radius: 2px;} | |
.button2 {background-color: #008CBA;} /* Blue */ | |
.button2 {border-radius: 2px;} | |
.button3 {background-color: #f44336;} /* Red */ | |
.button3 {border-radius: 2px;} | |
.button4 {background-color: #e7e7e7; color: black;} /* Gray */ | |
.button4 {border-radius: 2px;} | |
.button5 {background-color: #555555;} /* Black */ | |
.button5 {border-radius: 2px;} | |
table.list td:nth-child(2){ padding-left: 7px; } | |
table tr:nth-child(even) td:nth-child(even){ background: #BBBBBB; } | |
table tr:nth-child(odd) td:nth-child(odd){ background: #F2F2F2; } | |
table tr:nth-child(even) td:nth-child(odd){ background: #DDDDDD; } | |
table tr:nth-child(odd) td:nth-child(even){ background: #E5E5E5; } | |
div.column { width: 320px; float: left; } | |
div.first{ padding-right: 20px; border-right: 1px grey solid; } | |
div.second{ margin-left: 30px; } | |
table{ margin-left: 20px; } | |
-> | |
</style> | |
</head> | |
"@ | |
$secEventIDExpln = @" | |
<h2>Security Events</h2> | |
<table> | |
<tr> | |
<th>EventID</th> | |
<th>Event log</th> | |
<th>Description</th> | |
</tr> | |
<tr> | |
<td>4740</td> | |
<td>Security</td> | |
<td>Account Lockouts</td> | |
</tr> | |
<tr> | |
<td>4728,4732,4756</td> | |
<td>Security</td> | |
<td>User Added to Privileged Group</td> | |
</tr> | |
<tr> | |
<td>4735</td> | |
<td>Security</td> | |
<td>Security-Enabled group Modification</td> | |
</tr> | |
<tr> | |
<td>4624</td> | |
<td>Security</td> | |
<td>Successful User Account Login</td> | |
</tr> | |
<tr> | |
<td>4625</td> | |
<td>Security</td> | |
<td>Failed User Account Login </td> | |
</tr> | |
<tr> | |
<td>4648</td> | |
<td>Security</td> | |
<td>Account Login with Explicit Credentials </td> | |
</tr> | |
</table> | |
"@ | |
$hostname = $env:COMPUTERNAME | |
$pthXMLQueryLocal = @" | |
<QueryList> | |
<Query Id="0" Path="Security"> | |
<Select Path="Security"> | |
*[System[(Level=4 or Level=0) and (EventID=4624 or EventID=4625)]] | |
and | |
*[EventData[Data[@Name='LogonType'] and (Data='3')]] | |
and | |
*[EventData[Data[@Name='AuthenticationPackageName'] = 'NTLM']] | |
and | |
*[EventData[Data[@Name='TargetUserName'] != 'ANONYMOUS LOGON']] | |
and | |
*[EventData[Data[@Name='TargetDomainName'] != '$hostname']] | |
</Select> | |
</Query> | |
</QueryList> | |
"@ | |
$remoteDesktopLogon = @" | |
<QueryList> | |
<Query Id="0" Path="ForwardedEvents"> | |
<Select Path="ForwardedEvents"> | |
<!-- Collects Logon and Logoffs of RDP --> | |
<!-- Remote Desktop Protocol Connections --> | |
*[System[(Level=4 or Level=0) and (EventID=4624 or EventID=4634)]] | |
and | |
*[EventData[Data[@Name='LogonType']='10')]] | |
and | |
(*[EventData[Data[5]='10')]] | |
or | |
*[EventData[Data[@Name='AuthenticationPackageName'] = 'Negotiate']]) | |
</Select> | |
</Query> | |
</QueryList> | |
"@ | |
$noResults = @" | |
<p class="warning"> No events were found that match the specified selection criteria. </p> | |
"@ | |
function Invoke-DetectPTHL | |
{ | |
$winraw = Get-WinEvent -FilterXml $pthXMLQueryLocal | |
if($winraw) | |
{ | |
$justData = $winraw | Select-Object -Property Id, RecordId, LogName, ProcessId, ThreadId, MachineName, TimeCreated, TaskDisplayName, Message | |
$format = $justData | ConvertTo-Html -Head $header | |
} | |
else | |
{ | |
$format = $null | ConvertTo-Html -Head $header -Body $noResults | |
} | |
return $format | |
} | |
function Invoke-DetectRemoteLogon | |
{ | |
$winraw = Get-WinEvent -FilterXml $remoteDesktopLogon | |
if($winraw) | |
{ | |
$format = $winraw | ConvertTo-Html -Head $header | |
} | |
else | |
{ | |
$format = $null | ConvertTo-Html -Head $header -Body $noResults | |
} | |
return $format | |
} | |
$bodyExplain = @" | |
<ul class="menu"> | |
<li class="menu"><a class="menu" href="/query">Query</a></li> | |
<li class="menu"><a class="menu" id="active" href="/secevt">Security Event Logs</a></li> | |
<ul class="menu" style="float:right;list-style-type:none;"> | |
<li class="menu"><a class="menu" id="stop" href="/StopServer">Stop Server</a></li> | |
</ul> | |
</ul> | |
$secEventIDExpln | |
"@ | |
$bodyExplainQuery = @" | |
<ul class="menu"> | |
<li class="menu"><a class="menu" id="active" href="/query">Query</a></li> | |
<li class="menu"><a class="menu" href="/secevt">Security Event Logs</a></li> | |
<ul class="menu" style="float:right;list-style-type:none;"> | |
<li class="menu" ><a class="menu" id="stop" style="hove: " href="/StopServer">Stop Server</a></li> | |
</ul> | |
</ul> | |
$secEventIDExpln | |
<div class="tabs"> | |
<ul> | |
<li class="tab"> | |
<input type="radio" name="tabs" id="tab1" checked=""> | |
<label for="tab1">Query Events</label> | |
<div id="tab-content1" class="tab-content"> | |
<h2>Query Event</h2> | |
<span>Query a single event id or multiple deliminated by a "|" i.e. (4648|4624)</span> | |
<form method="get" target="queryEvent"> | |
EventID: | |
<input type="text" name="EventID" value=""><br> | |
<input type="submit" value="Submit" class="button button1"> | |
</form> | |
<iframe id="form-iframe" name="queryEvent" onload="AdjustIframeHeightOnLoad1()"></iframe> | |
<script type="text/javascript"> | |
function AdjustIframeHeightOnLoad1() { document.getElementById("form-iframe").style.height = document.getElementById("form-iframe").contentWindow.document.body.scrollHeight + "px"; } | |
function AdjustIframeHeight1(i) { document.getElementById("form-iframe").style.height = parseInt(i) + "px"; } | |
</script> | |
<br> | |
</div> | |
</li> | |
<li class="tab"> | |
<input type="radio" name="tabs" id="tab2"> | |
<label for="tab2">PTH Detection</label> | |
<div id="tab-content2" class="tab-content"> | |
<h2>Pass The Hash</h2> | |
<form method="get" target="queryEvent2"> | |
<input type="hidden" name="IOC" value="pth"><br> | |
<input type="submit" value="PTH Query" class="button button1"> | |
</form> | |
<iframe id="form-iframe2" name="queryEvent2" onload="AdjustIframeHeightOnLoad2()"></iframe> | |
<script type="text/javascript"> | |
function AdjustIframeHeightOnLoad2() { document.getElementById("form-iframe2").style.height = document.getElementById("form-iframe2").contentWindow.document.body.scrollHeight + "px"; } | |
function AdjustIframeHeight2(i) { document.getElementById("form-iframe2").style.height = parseInt(i) + "px"; } | |
</script> | |
</div> | |
</li> | |
<li class="tab"> | |
<input type="radio" name="tabs" id="tab3"> | |
<label for="tab3">Remote Desktop Logon Detection</label> | |
<div id="tab-content3" class="tab-content"> | |
<h2>Remote Logon Detection</h2> | |
<form method="get" target="queryEvent3"> | |
<input type="hidden" name="IOC" value="remotelogon"><br> | |
<input type="submit" value="Remote Logon Query" class="button button1"> | |
</form> | |
<iframe id="form-iframe3" name="queryEvent3" onload="AdjustIframeHeightOnLoad3()"></iframe> | |
<script type="text/javascript"> | |
function AdjustIframeHeightOnLoad3() { document.getElementById("form-iframe3").style.height = document.getElementById("form-iframe3").contentWindow.document.body.scrollHeight + "px"; } | |
function AdjustIframeHeight3(i) { document.getElementById("form-iframe3").style.height = parseInt(i) + "px"; } | |
</script> | |
</div> | |
</li> | |
</ul> | |
</div> | |
"@ | |
function Get-WindowsSecurityEvents | |
{ | |
$winraw = Get-EventLog -LogName security -Newest 100 | | |
Where-Object -FilterScript { | |
$_.EventID -match '4740|4728|4732|4756|4735|4625|4624|4648' | |
} | | |
Select-Object -Property TimeGenerated, Index, EventID, Source, MachineName, EntryType, Message | |
$format = $winraw | ConvertTo-Html -Head $header -Body $bodyExplain | |
return $format | |
} | |
function Query-WindowsSecurityEvents | |
{ | |
param($eventid) | |
if($eventid -eq $null -or '') | |
{ | |
$format = $null | ConvertTo-Html -Head $header -Body $bodyExplainQuery | |
return $format | |
} | |
else | |
{ | |
$winraw = Get-EventLog -LogName security -Newest 1000 | | |
Where-Object -FilterScript { | |
$_.EventID -match $eventid | |
} | | |
Select-Object -Property TimeGenerated, Index, EventID, Source, MachineName, EntryType, Message | |
$format = $winraw | ConvertTo-Html -Head $header -Body $bodyExplainQuery | |
return $format | |
} | |
} | |
function Query-WindowsSecurityEventsRaw | |
{ | |
param($eventid) | |
if($eventid -eq $null -or '') | |
{ | |
$format = $null | ConvertTo-Html -Head $header | |
return $format | |
} | |
else | |
{ | |
$winraw = Get-EventLog -LogName security -Newest 1000 | | |
Where-Object -FilterScript { | |
$_.EventID -match $eventid | |
} | | |
Select-Object -Property TimeGenerated, Index, EventID, Source, MachineName, EntryType, Message | |
$format = $winraw | ConvertTo-Html -Head $header | |
return $format | |
} | |
} | |
function Stop-Server | |
{ | |
$listener.Stop() | |
} | |
$routes = @{ | |
'/' = { | |
return Query-WindowsSecurityEvents | |
} | |
'/secevt' = { | |
return Get-WindowsSecurityEvents | |
} | |
'/query' = { | |
return Query-WindowsSecurityEvents | |
} | |
'/queryRaw' = { | |
return Query-WindowsSecurityEventsRaw | |
} | |
'/StopServer' = { | |
Stop-Server | |
} | |
} | |
$url = 'http://localhost:8088/' | |
$listener = New-Object -TypeName System.Net.HttpListener | |
$listener.Prefixes.Add($url) | |
$listener.Start() | |
Write-Host "Listening at $url..." | |
try | |
{ | |
while ($listener.IsListening) | |
{ | |
$context = $listener.GetContext() | |
$requestUrl = $context.Request.Url | |
$response = $context.Response | |
Write-Host '' | |
Write-Host "> $requestUrl" | |
$localPath = $requestUrl.LocalPath | |
$route = $routes.Get_Item($requestUrl.LocalPath) | |
Write-Host $requestUrl | |
if ($route -eq $null) | |
{ | |
$response.StatusCode = 404 | |
} | |
else | |
{ | |
if($requestUrl -match 'EventId') | |
{ | |
Write-Verbose -Message 'MATCH!!' | |
$found = [regex]::Match($requestUrl,'EventID\=((\d+\|+.*)|(\d+))').Value | |
Write-Host 'Found: ' $found | |
$eid = [regex]::Match($found,'((\d+\|+.*)|(\d+))').Value | |
Write-Host 'Querying ID: ' + $eid | |
$content = Query-WindowsSecurityEventsRaw -eventid $eid | |
$buffer = [System.Text.Encoding]::UTF8.GetBytes($content) | |
$response.ContentLength64 = $buffer.Length | |
$response.OutputStream.Write($buffer, 0, $buffer.Length) | |
} | |
elseif($requestUrl -match 'IOC\=pth') | |
{ | |
Write-Verbose -Message 'MATCH!!' | |
$found = [regex]::Match($requestUrl,'IOC\=(pth)').Value | |
Write-Host 'Found: ' $found | |
$eid = [regex]::Match($found,'(pth)').Value | |
Write-Host 'Querying for Pass The Hash Events' | |
$content = Invoke-DetectPTHL | |
$buffer = [System.Text.Encoding]::UTF8.GetBytes($content) | |
$response.ContentLength64 = $buffer.Length | |
$response.OutputStream.Write($buffer, 0, $buffer.Length) | |
} | |
elseif($requestUrl -match 'IOC\=remotelogon') | |
{ | |
Write-Verbose -Message 'MATCH!!' | |
$found = [regex]::Match($requestUrl,'IOC\=(remotelogon)').Value | |
Write-Host 'Found: ' $found | |
$eid = [regex]::Match($found,'(remotelogon)').Value | |
Write-Host 'Querying for Remote Desktop Logon Events' | |
$content = Invoke-DetectRemoteLogon | |
$buffer = [System.Text.Encoding]::UTF8.GetBytes($content) | |
$response.ContentLength64 = $buffer.Length | |
$response.OutputStream.Write($buffer, 0, $buffer.Length) | |
} | |
else | |
{ | |
$content = & $route | |
$buffer = [System.Text.Encoding]::UTF8.GetBytes($content) | |
$response.ContentLength64 = $buffer.Length | |
$response.OutputStream.Write($buffer, 0, $buffer.Length) | |
} | |
} | |
$response.Close() | |
$responseStatus = $response.StatusCode | |
Write-Host "< $responseStatus" | |
} | |
} | |
catch | |
{ | |
$listener.Stop() | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Work in progress...
How To: