通过安全日志读取WFP防火墙放行日志
之前的文档中,描写了如何对WFP防火墙进行操作以及如何在防火墙日志中读取被防火墙拦截网络通讯的日志。这边文档,着重描述如何读取操作系统中所有被放行的网络通信行为。读取系统中放行的网络通信行为日志,在win10之后的操作系统上,也可以通过前一篇提到的读取阻断日志的方式进行读取(以FWPM_NET_EVENT0.type字段区分),但是在较老的系统中却不支持直接读取。为了保持系统兼容性,可以通过读取操
前言
之前的文档中,描写了如何对WFP防火墙进行操作以及如何在防火墙日志中读取被防火墙拦截网络通讯的日志。这边文档,着重描述如何读取操作系统中所有被放行的网络通信行为。
读取系统中放行的网络通信行为日志,在win10之后的操作系统上,也可以通过前一篇提到的读取阻断日志的方式进行读取(以FWPM_NET_EVENT0.type字段区分),但是在较老的系统中却不支持直接读取。为了保持系统兼容性,可以通过读取操作系统安全日志(EventId:5156)的方式进行网络通信日志的采集。
需要注意的坑点
- 查询放行日志时需要注意,每个网络通信行为在日志中只会出现一条放行记录,对应的筛选器ID,只会是首次对其进行审计的过滤器ID。因此,如果有其他权重更高的子层对网络连接进行了审计时,就无法通过筛选器ID匹配的方式获取。如果有这方面需求的话,解决方法只能是尽可能将自身子层的权重设为最高。
- 网络日志中读取连入行为时,WIN10/2016/2019源IP和目的IP字段与其它更早的操作系统相反,需要特殊处理。连出行为无异常。
开启审计
采用读取安全日志的方式进行网络事件获取,首先需要在系统中开启审计功能。在代码里面也有多种方式可以开启,之后会单开一篇文档进行描述,在这里先手动开启。
-
打开本地安全策略(开始——运行——secpol.msc),依次打开:安全设置——本地策略——审核策略如图
 -
在右侧窗口中打开 审核对象 标签页,勾选 “成功” 复选框后,点击保存,即可开启网络访问的审计功能
 -
右键单击 “我的电脑”——“管理”——“计算机管理”——“系统工具”——“事件查看器”——“Windows日志”——“安全”中,查看5156日志即可。

网络通信日志默认情况下是开启状态,为了以防万一,每次获取之前需要使用代码开启一次。使用代码的开启方式下次单开文档分享。
使用WMI方式进行查询
使用ReadEventLog进行查询
优点:兼容性高,可支持XP/2003操作系统。读取性能高。
缺点:无法做过滤,在大量日志中提取少量日志时效率较低
使用ReadEventLog读取Windows的安全日志只需要三步即可,1、打开EventLog句柄;2、使用ReadEventLog循环读取日志;3、关闭EventLog句柄。具体API描述如下。
打开EventLog句柄
HANDLE OpenEventLog(
LPCSTR lpUNCServerName,
LPCSTR lpSourceName
);
- 输入参数
- lpUNCServerName:远程服务器的名称。读取本地的话传入NULL即可。
- lpSourceName:日志名称。这里读取安全日志传入“Serurity”。其他对应值:系统日志“System”,应用程序日志“Application”
- 输出参数
- 返回日志读取句柄。在ReadEventLog中使用,需要调用CloseEventLog手动关闭。
读取日志
BOOL ReadEventLog(
HANDLE hEventLog,
DWORD dwReadFlags,
DWORD dwRecordOffset,
LPVOID lpBuffer,
DWORD nNumberOfBytesToRead,
DWORD *pnBytesRead,
DWORD *pnMinNumberOfBytesNeeded
);
-
输入参数
- hEventLog:需要读取日志的句柄,就是刚才OpenEventLog返回的那个
- dwReadFlags:读取标志,可以选择从指定偏移读取(EVENTLOG_SEEK_READ)或按顺序读取(EVENTLOG_SEQUENTIAL_READ),也可以指定正序读取(EVENTLOG_FORWARDS_READ)或倒序读取(EVENTLOG_BACKWARDS_READ)
- dwRecordOffset:当dwReadFlags中包含EVENTLOG_SEEK_READ时有效,表示开始的位置
- lpBuffer:分配的缓冲区,由外部划分内存
- nNumberOfBytesToRead:lpBuffer缓冲区的大小
- pnBytesRead:返回接收字节数
- pnMinNumberOfBytesNeeded:返回lpBuffer所需最小缓冲区大小。仅当lpBuffer过小时返回,可判断GetLastError()返回ERROR_INSUFFICIENT_BUFFER
时有效。
-
输出参数
- 正常执行返回非0值,失败后返回0
关闭EventLog句柄
BOOL CloseEventLog(
HANDLE hEventLog
);
- 输入参数
- hEventLog:事件句柄,由OpenEventLog返回
- 输出参数
- 成功与否,这玩意没啥好判断的。
关联结构体
typedef struct _EVENTLOGRECORD {
DWORD Length;
DWORD Reserved;
DWORD RecordNumber;
DWORD TimeGenerated;
DWORD TimeWritten;
DWORD EventID;
WORD EventType;
WORD NumStrings;
WORD EventCategory;
WORD ReservedFlags;
DWORD ClosingRecordNumber;
DWORD StringOffset;
DWORD UserSidLength;
DWORD UserSidOffset;
DWORD DataLength;
DWORD DataOffset;
} EVENTLOGRECORD, *PEVENTLOGRECORD;
- 参数说明
-
Length:当前结构体的长度,由于ReadEventLog是以内存块的方式返回,单次返回的内存块中可能包含多个,特别是为了节省资源,可能会在代码中刻意一次读取大量的EventLogRecord结构体。这些结构体在内存块中,就以Length参数作为分界线来进行分割
-
Reserved:保留,没有可以研究过用于啥
-
RecordNumber:日志序号,可配合ReadEventLog函数中的dwReadFlags参数和dwReadOffset参数设置读取的偏移地址
-
TimeGenerated:事件时间,转time_t就可以
-
TimeWrittern:日志写入时间,time_t格式
-
EventID:事件ID,最高两位代表严重性,第三位代表是否为系统事件,低16位代表在安全日志中可见的ID。
-
EventType:事件类型,
-
NumStrings:包含字符串数目
-
EventCategory:事件类别
-
ReservedFlags:保留
-
ClosingRecordNumber:保留
-
StringOffset:事件包含字符串起始地址的偏移。字符串按顺序依次在内存中存储,0长度的字符串代表结束。字符串的顺序,和在事件查看器中看到的顺序一致,如图
 -
UserSidLength/UserSidOffset: 用户SID的长度及偏移,与字符串类似
-
DataLength/DataOffset:数据部分的长度及偏移
-
参考代码
#include <iostream>
#include <windows.h>
#include <string>
#include <vector>
#define MAX_EVENTLOG_READONCE 2048
void ReadLogsByReadEventLogAPI()
{
LPBYTE pEventLogBuffer = new(std::nothrow) BYTE[MAX_EVENTLOG_READONCE];
if (pEventLogBuffer == NULL)
{
return;
}
ZeroMemory(pEventLogBuffer, MAX_EVENTLOG_READONCE);
HANDLE hEventLog = OpenEventLog(NULL, L"Security");
if (hEventLog == NULL)
{
delete[] pEventLogBuffer;
pEventLogBuffer = NULL;
return;
}
DWORD dwMemoryLen = MAX_EVENTLOG_READONCE;
DWORD dwReaded = 0, dwMiniMemoryNeeded = 0;
while (true)// -_-|||
{
BOOL isSucc = ReadEventLog(hEventLog, EVENTLOG_BACKWARDS_READ | EVENTLOG_SEQUENTIAL_READ, 0, pEventLogBuffer, dwMemoryLen, &dwReaded, &dwMiniMemoryNeeded);
if (isSucc == FALSE && GetLastError() == ERROR_INSUFFICIENT_BUFFER && dwMiniMemoryNeeded)
{
delete[] pEventLogBuffer;
pEventLogBuffer = NULL;
pEventLogBuffer = new (std::nothrow) BYTE[dwMiniMemoryNeeded];
if (pEventLogBuffer == NULL)
{
break;
}
ZeroMemory(pEventLogBuffer, dwMiniMemoryNeeded);
dwMemoryLen = dwMiniMemoryNeeded;
isSucc = ReadEventLog(hEventLog, EVENTLOG_BACKWARDS_READ | EVENTLOG_SEQUENTIAL_READ, 0, pEventLogBuffer, dwMemoryLen, &dwReaded, &dwMiniMemoryNeeded);
}
if (isSucc == FALSE)
{
break;
}
int pos = 0;
do
{
EVENTLOGRECORD* pTempRecord = (EVENTLOGRECORD*)(pEventLogBuffer + pos);
__time32_t occurTime = pTempRecord->TimeWritten;
int nEventId = pTempRecord->EventID & 0xffff;
if (nEventId != 5156)
{
pos = pos + pTempRecord->Length;
continue;
}
std::vector<std::wstring> vecEventParam;
LPBYTE pTempBuffer = (LPBYTE)pTempRecord + pTempRecord->StringOffset;
int strCount = 0;
while ((pTempBuffer < ((LPBYTE)pTempRecord + pTempRecord->Length)) && (strCount < pTempRecord->NumStrings))
{
int len = wcslen((wchar_t*)pTempBuffer);
if (len == 0)
{
break;
}
vecEventParam.push_back((wchar_t*)pTempBuffer);
pTempBuffer = pTempBuffer + (len + 1) * sizeof(wchar_t);
strCount++;
}
if (vecEventParam.size() > 10)
{
std::wcout << "\n\n=========================================" << std::endl;
std::wcout << L"Source:\t" << vecEventParam[3] << L"[" << vecEventParam[4] << L"]" << std::endl;
std::wcout << L"Destination:\t" << vecEventParam[5] << L"[" << vecEventParam[6] << L"]" << std::endl;
std::wcout << L"Protocol Code:\t" << vecEventParam[7] << std::endl;
std::wcout << L"Process Id:\t" << vecEventParam[0] << std::endl;
std::wcout << L"Process Name:\t" << vecEventParam[1] << std::endl;
std::wcout << "=========================================\n\n" << std::endl;
}
pos = pos + pTempRecord->Length;
} while (pos < dwReaded);
}
CloseEventLog(hEventLog);
delete[] pEventLogBuffer;
pEventLogBuffer = NULL;
}
输出截图

使用Evt系列API进行查询
优点:性能高,可自由配置过滤条件,方便在海量日志中检索,可读取的内容相当丰富
缺点:不支持xp/2003操作系统
Evt系列API涉及到的功能较多,如果只需要读取系统日志的话,只需要枚举、遍历、读取、关闭四步即可完成。
枚举当前日志
EVT_HANDLE EvtQuery(
EVT_HANDLE Session,
LPCWSTR Path,
LPCWSTR Query,
DWORD Flags
);
遍历日志,获取下一条
BOOL EvtNext(
EVT_HANDLE ResultSet,
DWORD EventsSize,
PEVT_HANDLE Events,
DWORD Timeout,
DWORD Flags,
PDWORD Returned
);
读取日志内容
BOOL EvtRender(
EVT_HANDLE Context,
EVT_HANDLE Fragment,
DWORD Flags,
DWORD BufferSize,
PVOID Buffer,
PDWORD BufferUsed,
PDWORD PropertyCount
);
- 输入参数
- Context:
- Fragment:
- Flags:
- BufferSize:
- Buffer:
- BufferUsed:
- PropertyCount:
- 输出参数
关闭枚举句柄
BOOL EvtClose(
EVT_HANDLE Object
);
实例代码
void parseEventXML(std::wstring wstrXML)
{
std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>> cv;
std::string utfXML = cv.to_bytes(wstrXML);
tinyxml2::XMLDocument rootXML;
if (rootXML.Parse(utfXML.c_str()) != tinyxml2::XML_SUCCESS)
{
return;
}
tinyxml2::XMLNode* rootNode = rootXML.FirstChild();
if (rootNode == NULL)
{
return;
}
std::string processId;
std::string processName;
std::string sourceAddress;
std::string sourcePort;
std::string destAddress;
std::string destPort;
std::string protocol;
tinyxml2::XMLNode* eventNode = rootXML.FirstChildElement("Event");
if (eventNode)
{
tinyxml2::XMLElement* eventDataNode = eventNode->FirstChildElement("EventData");
if (eventDataNode == NULL)
{
return;
}
tinyxml2::XMLElement* dataNode = eventDataNode->FirstChildElement("Data");
do
{
std::string strName = dataNode->Attribute("Name");
if (_stricmp(strName.c_str(), "ProcessID") == 0)
{
processId = dataNode->GetText();
}
else if (_stricmp(strName.c_str(), "Application") == 0)
{
processName = dataNode->GetText();
}
else if (_stricmp(strName.c_str(), "SourceAddress") == 0)
{
sourceAddress = dataNode->GetText();
}
else if (_stricmp(strName.c_str(), "SourcePort") == 0)
{
sourcePort = dataNode->GetText();
}
else if (_stricmp(strName.c_str(), "DestAddress") == 0)
{
destAddress = dataNode->GetText();
}
else if (_stricmp(strName.c_str(), "DestPort") == 0)
{
destPort = dataNode->GetText();
}
else if (_stricmp(strName.c_str(), "Protocol") == 0)
{
protocol = dataNode->GetText();
}
dataNode = dataNode->NextSiblingElement("Data");
} while (dataNode);
std::cout << "\n\n=========================================" << std::endl;
std::cout << "Source:\t" << sourceAddress << "[" << sourcePort << "]" << std::endl;
std::cout << "Destination:\t" << destAddress << "[" << destPort << "]" << std::endl;
std::cout << "Protocol Code:\t" << protocol << std::endl;
std::cout << "Process Id:\t" << processId << std::endl;
std::cout << "Process Name:\t" << processName << std::endl;
std::cout << "=========================================\n\n" << std::endl;
}
}
void ReadLogsByEvtAPI()
{
DWORD dwEventLogBufferLen = MAX_EVENTLOG_READONCE;
LPBYTE pEventLogBuffer = new(std::nothrow) BYTE[MAX_EVENTLOG_READONCE];
if (pEventLogBuffer == NULL)
{
return;
}
ZeroMemory(pEventLogBuffer, MAX_EVENTLOG_READONCE);
std::wstring wstrQuery = std::wstring(
L"<QueryList>"
L" <Query>"
L" <Select>Event/System[EventID=5156]</Select>"
L" </Query>"
L"</QueryList>");
EVT_HANDLE hResult = EvtQuery(NULL, L"Security", wstrQuery.c_str(), EvtQueryChannelPath | EvtQueryReverseDirection);
if (hResult == NULL)
{
delete[] pEventLogBuffer;
pEventLogBuffer = NULL;
return;
}
while (true) // -_-
{
EVT_HANDLE hEventArrs[MAX_PATH] = { 0 };
DWORD dwReturnEvents = 0;
BOOL isSucc = EvtNext(hResult, MAX_PATH, hEventArrs, INFINITE, 0, &dwReturnEvents);
if (isSucc && dwReturnEvents)
{
for (int eventPos = 0; eventPos < dwReturnEvents; eventPos++)
{
DWORD dwBufferUsed = 0;
DWORD dwPropertyCount = 0;
int nBufferSize = dwEventLogBufferLen;
ZeroMemory(pEventLogBuffer, nBufferSize);
isSucc = EvtRender(NULL, hEventArrs[eventPos], EvtRenderEventXml, nBufferSize, pEventLogBuffer, &dwBufferUsed, &dwPropertyCount);
if (isSucc == FALSE && GetLastError() == ERROR_INSUFFICIENT_BUFFER)
{
delete[] pEventLogBuffer;
pEventLogBuffer = NULL;
dwEventLogBufferLen = dwBufferUsed + 2;
pEventLogBuffer = new(std::nothrow) BYTE[dwEventLogBufferLen];
if (pEventLogBuffer == NULL)
{
break;
}
nBufferSize = dwEventLogBufferLen;
ZeroMemory(pEventLogBuffer, dwEventLogBufferLen);
isSucc = EvtRender(NULL, hEventArrs[eventPos], EvtRenderEventXml, nBufferSize, pEventLogBuffer, &dwBufferUsed, &dwPropertyCount);
}
if (isSucc)
{
std::wstring strEventXML = std::wstring((wchar_t*)pEventLogBuffer);
parseEventXML(strEventXML);
}
}
}
}
if (pEventLogBuffer)
{
delete[] pEventLogBuffer;
pEventLogBuffer = NULL;
}
}
输出截图

备注
实例代码中解析xml使用的是tinyxml2库,对应github地址为:https://github.com/leethomason/tinyxml2/tree/master
当前最新版(9.0.0)版本下载地址:https://download.csdn.net/download/QQ1113130712/88235095
接下来我将给各位同学划分一张学习计划表!
学习计划
那么问题又来了,作为萌新小白,我应该先学什么,再学什么?
既然你都问的这么直白了,我就告诉你,零基础应该从什么开始学起:
阶段一:初级网络安全工程师
接下来我将给大家安排一个为期1个月的网络安全初级计划,当你学完后,你基本可以从事一份网络安全相关的工作,比如渗透测试、Web渗透、安全服务、安全分析等岗位;其中,如果你等保模块学的好,还可以从事等保工程师。
综合薪资区间6k~15k
1、网络安全理论知识(2天)
①了解行业相关背景,前景,确定发展方向。
②学习网络安全相关法律法规。
③网络安全运营的概念。
④等保简介、等保规定、流程和规范。(非常重要)
2、渗透测试基础(1周)
①渗透测试的流程、分类、标准
②信息收集技术:主动/被动信息搜集、Nmap工具、Google Hacking
③漏洞扫描、漏洞利用、原理,利用方法、工具(MSF)、绕过IDS和反病毒侦察
④主机攻防演练:MS17-010、MS08-067、MS10-046、MS12-20等
3、操作系统基础(1周)
①Windows系统常见功能和命令
②Kali Linux系统常见功能和命令
③操作系统安全(系统入侵排查/系统加固基础)
4、计算机网络基础(1周)
①计算机网络基础、协议和架构
②网络通信原理、OSI模型、数据转发流程
③常见协议解析(HTTP、TCP/IP、ARP等)
④网络攻击技术与网络安全防御技术
⑤Web漏洞原理与防御:主动/被动攻击、DDOS攻击、CVE漏洞复现
5、数据库基础操作(2天)
①数据库基础
②SQL语言基础
③数据库安全加固
6、Web渗透(1周)
①HTML、CSS和JavaScript简介
②OWASP Top10
③Web漏洞扫描工具
④Web渗透工具:Nmap、BurpSuite、SQLMap、其他(菜刀、漏扫等)
那么,到此为止,已经耗时1个月左右。你已经成功成为了一名“脚本小子”。那么你还想接着往下探索吗?
阶段二:中级or高级网络安全工程师(看自己能力)
综合薪资区间15k~30k
7、脚本编程学习(4周)
在网络安全领域。是否具备编程能力是“脚本小子”和真正网络安全工程师的本质区别。在实际的渗透测试过程中,面对复杂多变的网络环境,当常用工具不能满足实际需求的时候,往往需要对现有工具进行扩展,或者编写符合我们要求的工具、自动化脚本,这个时候就需要具备一定的编程能力。在分秒必争的CTF竞赛中,想要高效地使用自制的脚本工具来实现各种目的,更是需要拥有编程能力。
零基础入门的同学,我建议选择脚本语言Python/PHP/Go/Java中的一种,对常用库进行编程学习
搭建开发环境和选择IDE,PHP环境推荐Wamp和XAMPP,IDE强烈推荐Sublime;
Python编程学习,学习内容包含:语法、正则、文件、 网络、多线程等常用库,推荐《Python核心编程》,没必要看完
用Python编写漏洞的exp,然后写一个简单的网络爬虫
PHP基本语法学习并书写一个简单的博客系统
熟悉MVC架构,并试着学习一个PHP框架或者Python框架 (可选)
了解Bootstrap的布局或者CSS。
阶段三:顶级网络安全工程师
如果你对网络安全入门感兴趣,那么你需要的话可以点击这里👉网络安全重磅福利:入门&进阶全套282G学习资源包免费分享!

学习资料分享
当然,只给予计划不给予学习资料的行为无异于耍流氓,这里给大家整理了一份【282G】的网络安全工程师从入门到精通的学习资料包,可点击下方二维码链接领取哦。

更多推荐
所有评论(0)