Skip to content

Instantly share code, notes, and snippets.

@watson0x90
Last active March 7, 2016 15:30
Show Gist options
  • Save watson0x90/f04aba16c41f028dcff5 to your computer and use it in GitHub Desktop.
Save watson0x90/f04aba16c41f028dcff5 to your computer and use it in GitHub Desktop.
Query Windows Security Event Log via PowerShell HTTP Server
#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()
}
@watson0x90
Copy link
Author

Work in progress...
How To:

  1. Download script to localhost
  2. Start PowerShell console with local admin privileges
  3. Run the script
  4. Navigate in a browser to http://localhost:8088
  5. Don't whine and live with the interface...
  6. Click "Stop Server" in browser to shutdown the script. (Top right big red button.)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment