성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Java - How to use the Foreign Funct...
[정성태] 제가 큰 실수를 했군요. ^^; Delegate를 통한 Bein...
[정성태] Working with Rust Libraries from C#...
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
[양승조
] var대신 dinamic으로 선언해서 해결은 했습니다. 맞는 해...
[양승조
] 또 막혔습니다. ㅠㅠ var list = props[i].Ge...
[양승조
] 아. 감사합니다. 어제는 안됐던것 같은데....정신을 차려야겠네...
[정성태] "props[i].GetValue(props[i])" 코드에서 ...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>C# - IPGlobalProperties를 이용해 netstat처럼 사용 중인 Socket 목록 구하는 방법</h1> <p> IPGlobalProperties를 이용하면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > IPGlobalProperties Class ; <a target='tab' href='https://learn.microsoft.com/en-us/dotnet/api/system.net.networkinformation.ipglobalproperties'>https://learn.microsoft.com/en-us/dotnet/api/system.net.networkinformation.ipglobalproperties</a> </pre> <br /> <a target='tab' href='https://www.sysnet.pe.kr/2/0/1826#netstat'>netstat</a>를 이용한 출력 결과를 코드로 가져오는 것이 가능합니다.<br /> <br /> 예를 들어, "<a target='tab' href='https://learn.microsoft.com/en-us/windows-server/administration/windows-commands/netstat'>netstat</a> -ano | findstr LISTEN"처럼 현재 열려 있는 TCP 서버 소켓을 다음과 같은 코드로 나열하는 것이 가능합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > { var props = IPGlobalProperties.GetIPGlobalProperties(); var listeners = props.<span style='color: blue; font-weight: bold'>GetActiveTcpListeners</span>(); // Listen 중인 TCP 소켓을 열거 foreach (var item in listeners) { Console.WriteLine(item); } } </pre> <br /> 이것을 이용하면, 특정 서비스로의 연결을 가지고 있는지 테스트하는 것도 가능합니다. 예를 들어, 현재 머신에서 SQL Server (1433)에 대한 연결이 있는지,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > { var props = IPGlobalProperties.GetIPGlobalProperties(); var listeners = props.<span style='color: blue; font-weight: bold'>GetActiveTcpConnections</span>(); foreach (var item in listeners) { <span style='color: blue; font-weight: bold'>if (item.RemoteEndPoint.Port == 1433)</span> { Console.WriteLine($"{item.LocalEndPoint}-{item.RemoteEndPoint}"); } } } /* 192.168.100.20:42621-192.168.100.50:1433 */ </pre> <br /> 저런 식으로 확인할 수도 있습니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> IPGlobalProperties의 구현 코드는 윈도우의 경우 GetTcpTable Win32 API를 호출하게 됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > GetTcpTable function (iphlpapi.h) ; <a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/iphlpapi/nf-iphlpapi-gettcptable'>https://learn.microsoft.com/en-us/windows/win32/api/iphlpapi/nf-iphlpapi-gettcptable</a> </pre> <br /> 이 함수의 사용법은 아래의 글에서 잘 설명하고 있는데요,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Getting active TCP/UDP connections on a box ; <a target='tab' href='https://www.codeproject.com/Articles/4298/Getting-active-TCP-UDP-connections-on-a-box'>https://www.codeproject.com/Articles/4298/Getting-active-TCP-UDP-connections-on-a-box</a> </pre> <br /> 재미있는 것은, GetTcpTable은 해당 소켓이 속한 Process ID를 가져오지는 않습니다. 그런데 위의 글에 보면, 문서화되지 않은 AllocateAndGetTcpExTableFromStack API의 경우 Process ID를 가져올 수 있다고 하는데요, 현재 시점(2024-01-02)에는 다음과 같이 문서화된 상태입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > AllocateAndGetTcpExTableFromStack function (iphlpapi.h) ; <a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/iphlpapi/nf-iphlpapi-allocateandgettcpextablefromstack'>https://learn.microsoft.com/en-us/windows/win32/api/iphlpapi/nf-iphlpapi-allocateandgettcpextablefromstack</a> </pre> <br /> 단지, 도움말에도 나오지만 지원이 끊길 예정이고, 대신 <a target='tab' href='https://learn.microsoft.com/en-us/windows/win32/api/iphlpapi/nf-iphlpapi-getextendedtcptable'>GetExtendedTcpTable</a> 함수를 사용하라고 나옵니다.<br /> <br /> 그런데 사실 닷넷에서도 IPv6 정보에 대해서는 GetExtendedTcpTable을 이용해 조회를 하고 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > /// 닷넷 소스 코드 /// Gets the active TCP connections. Uses the native GetTcpTable API. private static unsafe List<SystemTcpConnectionInformation> GetAllTcpConnections() { uint size = 0; uint result; List<SystemTcpConnectionInformation> tcpConnections = new List<SystemTcpConnectionInformation>(); // Check if it supports IPv4 for IPv6 only modes. if (Socket.OSSupportsIPv4) { // ...[생략]... result = Interop.IpHlpApi.<span style='color: blue; font-weight: bold'>GetTcpTable</span>(buffer, &size, order: true); // ...[생략]... } if (Socket.OSSupportsIPv6) { // ...[생략]... result = Interop.IpHlpApi.<span style='color: blue; font-weight: bold'>GetExtendedTcpTable</span>(IntPtr.Zero, &size, order: true, (uint)AddressFamily.InterNetworkV6, // ...[생략]... } return tcpConnections; } </pre> <br /> 따라서, IPv4에 대해서도 간단한 소스코드 변경만으로 GetExtendedTcpTable을 지원할 수 있었을 것이고, 자연스럽게 Process ID를 구할 수 있었을 텐데도 관련 코드는 누락이 된 상태입니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 결국, 소켓에 대한 연관 프로세스를 알고 싶다면 netstat를 통해 우회하던가,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > c:\temp> <span style='color: blue; font-weight: bold'>netstat -ano | findstr LISTEN</span> TCP 0.0.0.0:80 0.0.0.0:0 LISTENING <span style='color: blue; font-weight: bold'>4</span> TCP 0.0.0.0:135 0.0.0.0:0 LISTENING <span style='color: blue; font-weight: bold'>1824</span> TCP 0.0.0.0:445 0.0.0.0:0 LISTENING <span style='color: blue; font-weight: bold'>4</span> TCP 0.0.0.0:1433 0.0.0.0:0 LISTENING <span style='color: blue; font-weight: bold'>7884</span> ...[생략]... </pre> <br /> 아니면 직접 GetExtendedTcpTable API를 사용하는 코드를 작성해야 합니다. 사실 이에 대한 포팅이 매우 쉬운데요, 닷넷 소스코드의 전체적인 구조를 그대로 베끼면서 GetExtendedTcpTable을 호출하도록 변경하는 정도만 신경 쓰면 되므로, 대충 다음과 같이 구현하는 것이 가능합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > static unsafe List<SystemTcpConnectionInformation> GetAllTcpConnections(AF_CLASS afClass, TCP_TABLE_CLASS tableClass) { uint size = 0; uint result; List<SystemTcpConnectionInformation> tcpConnections = new List<SystemTcpConnectionInformation>(); // Check if it supports IPv4 for IPv6 only modes. if (Socket.OSSupportsIPv4 && afClass == AF_CLASS.AF_INET) { // Get the buffer size needed. result = GetExtendedTcpTable(IntPtr.Zero, out size, true, afClass, tableClass, 0); while (result == <a target='tab' href='https://www.sysnet.pe.kr/2/0/13206#17132'>ERROR_INSUFFICIENT_BUFFER</a>) { // Allocate the buffer and get the TCP table. IntPtr buffer = Marshal.AllocHGlobal((int)size); try { result = <span style='color: blue; font-weight: bold'>GetExtendedTcpTable</span>(buffer, out size, true, afClass, tableClass, 0); if (result == ERROR_SUCCESS) { var span = new ReadOnlySpan<byte>((byte*)buffer, (int)size); // The table info just gives us the number of rows. ref readonly MibTcpTableOwnerPid tcpTableInfo = ref MemoryMarshal.AsRef<MibTcpTableOwnerPid>(span); if (tcpTableInfo.numberOfEntries > 0) { // Skip over the tableinfo to get the inline rows. span = span.Slice(sizeof(MibTcpTableOwnerPid)); for (int i = 0; i < tcpTableInfo.numberOfEntries - 1; i++) { SystemTcpConnectionInformation item = new SystemTcpConnectionInformation(in MemoryMarshal.AsRef<MibTcpRowOwnerPid>(span)); tcpConnections.Add(item); span = span.Slice(sizeof(MibTcpRowOwnerPid)); } } } } finally { Marshal.FreeHGlobal(buffer); } } // If we don't have any ipv4 interfaces detected, just continue. if (result != ERROR_SUCCESS && result != ERROR_NO_DATA) { throw new NetworkInformationException((int)result); } } if (Socket.OSSupportsIPv6 && afClass == AF_CLASS.AF_INET6) { // Get the buffer size needed. size = 0; result = GetExtendedTcpTable(IntPtr.Zero, out size, true, afClass, tableClass, 0); while (result == ERROR_INSUFFICIENT_BUFFER) { // Allocate the buffer and get the TCP table. IntPtr buffer = Marshal.AllocHGlobal((int)size); try { result = <span style='color: blue; font-weight: bold'>GetExtendedTcpTable</span>(buffer, out size, true, afClass, tableClass, 0); if (result == ERROR_SUCCESS) { var span = new ReadOnlySpan<byte>((byte*)buffer, (int)size); // The table info just gives us the number of rows. ref readonly MibTcp6TableOwnerPid tcpTable6OwnerPid = ref MemoryMarshal.AsRef<MibTcp6TableOwnerPid>(span); if (tcpTable6OwnerPid.numberOfEntries > 0) { // Skip over the tableinfo to get the inline rows. span = span.Slice(sizeof(MibTcp6TableOwnerPid)); for (int i = 0; i < tcpTable6OwnerPid.numberOfEntries; i++) { tcpConnections.Add(new SystemTcpConnectionInformation(in MemoryMarshal.AsRef<MibTcp6RowOwnerPid>(span))); // We increment the pointer to the next row. span = span.Slice(sizeof(MibTcp6RowOwnerPid)); } } } } finally { Marshal.FreeHGlobal(buffer); } } // If we don't have any ipv6 interfaces detected, just continue. if (result != ERROR_SUCCESS && result != ERROR_NO_DATA) { throw new NetworkInformationException((int)result); } } return tcpConnections; } </pre> <br /> 그다음 이렇게 사용해 주면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > { var props = IPGlobalPropertiesExtension.GetTcpIPv4Listeners(); foreach (var item in props) { Console.WriteLine(item); } } { var props = IPGlobalPropertiesExtension.GetTcpIPv6Listeners(); foreach (var item in props) { Console.WriteLine(item); } } </pre> <br /> 다음과 같은 출력 결과를 얻을 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 0.0.0.0:80 Listen (pid:4) 0.0.0.0:135 Listen (pid:1824) 0.0.0.0:445 Listen (pid:4) 0.0.0.0:1433 Listen (pid:7884) ...[생략]... [::]:80 Listen (pid:4) [::]:135 Listen (pid:1824) [::]:445 Listen (pid:4) [::]:1433 Listen (pid:7884) ...[생략]... </pre> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=2129&boardid=331301885'>첨부 파일은 이 글의 소스코드를 포함</a>합니다.)<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
2055
(왼쪽의 숫자를 입력해야 합니다.)