Microsoft MVP성태의 닷넷 이야기
닷넷: 2206. C# - TCP KeepAlive의 서버 측 구현 [링크 복사], [링크+제목 복사],
조회: 10300
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 6개 있습니다.)
.NET Framework: 487. Socket.Receive 메서드의 SocketFlags.Peek 동작을 이용해 소켓 연결 유무를 확인?
; https://www.sysnet.pe.kr/2/0/1824

.NET Framework: 488. TCP 소켓 연결의 해제를 알 수 있는 방법
; https://www.sysnet.pe.kr/2/0/1825

닷넷: 2204. C# - TCP KeepAlive에 새로 추가된 Retry 옵션
; https://www.sysnet.pe.kr/2/0/13531

닷넷: 2206. C# - TCP KeepAlive의 서버 측 구현
; https://www.sysnet.pe.kr/2/0/13533

Windows: 255. (디버거의 영향 등으로) 대상 프로세스가 멈추면 Socket KeepAlive로 연결이 끊길까요?
; https://www.sysnet.pe.kr/2/0/13546

Windows: 256. C# - Server socket이 닫히면 Accept 시켰던 자식 소켓이 닫힐까요?
; https://www.sysnet.pe.kr/2/0/13550




C# - TCP KeepAlive의 서버 측 구현

지난 이야기에서는,

C# - TCP KeepAlive에 새로 추가된 Retry 옵션
; https://www.sysnet.pe.kr/2/0/13531

클라이언트로 예제를 사용했는데요, 사실 현업으로 따지자면 서버 측에서의 구현이 더 일반적일 수 있습니다. 왜냐하면, 서버는 다수의 클라이언트와 연결을 맺는 것이므로 쓸데없는 연결이 남게 되면 자칫 자원 고갈의 문제로 이어질 수 있기 때문입니다. 따라서 끊긴 클라이언트를 가능하면 빠르게 인지하는 것이 서버 입장에서는 더 필요할 수밖에 없습니다.

일단, (클라이언트와는 달리) 서버는 그나마 자유롭게 고를 수 있으므로 Windows Server 2019 이상으로 가정한다면 TCP 레벨에 추가된 3가지 옵션(TcpKeepAliveTime, TcpKeepAliveInterval, TcpKeepAliveRetryCount)을 걱정 없이 사용할 수 있을 것입니다.

적용 방법은 간단한데요, Server 소켓에만 적용해 주면 이후 Accept한 클라이언트 소켓에 서버의 설정이 상속되므로 다음과 같이 간단하게 KeepAlive 설정을 할 수 있습니다.

int port = 18500;
Socket listenSocket = new Socket(AddressFamily.InterNetwork,
                                SocketType.Stream,
                                ProtocolType.Tcp);


if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows) == false
    || Environment.OSVersion.Version < new Version(10, 0, 15063))
{
    throw new NotSupportedException();
}

listenSocket.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.KeepAlive, true);
listenSocket.SetSocketOption(SocketOptionLevel.Tcp, (SocketOptionName)3, 1);
listenSocket.SetSocketOption(SocketOptionLevel.Tcp, (SocketOptionName)17, 3);
listenSocket.SetSocketOption(SocketOptionLevel.Tcp, (SocketOptionName)16, 10);

이후, 실제로 (VM을 Pause 시킨 순간의) 연결 끊김을 알아내기 위해 Ping 코드를 추가해 다음과 같이 테스트 서버를 완성할 수 있습니다.

IPEndPoint ep = new IPEndPoint(IPAddress.Any, port);
listenSocket.Bind(ep);
listenSocket.Listen(5);

while (true)
{
    Socket clientSocket = listenSocket.Accept();
    Log($"{clientSocket} connected.");
    bool pingEnabled = true;

    Task.Run(() =>
    {
        bool connected = true;
        IPEndPoint? ep = clientSocket.RemoteEndPoint as IPEndPoint;
        if (ep == null)
        {
            return;
        }

        while (pingEnabled)
        {
            Ping ping = new Ping();

            PingOptions options = new PingOptions();
            options.DontFragment = true;

            string data = "test";
            byte[] buffer = ASCIIEncoding.ASCII.GetBytes(data);
            int timeout = 300;

                    
            PingReply reply = ping.Send(ep.Address, timeout, buffer, options);
            bool replied = reply.Status == IPStatus.Success;

            if (connected != replied)
            {
                connected = replied;
                Log($"Status changed to {reply.Status}");
            }

            Thread.Sleep(32);
        }
    });

    Task.Run(() =>
    {
        Log("SendData");
        clientSocket.Send(new byte[4] { 1, 2, 3, 4 });

        Log("Wait for receiving");

        try
        {
            clientSocket.Receive(new byte[4]);
            Log($"Received");
        }
        catch (Exception e)
        {
            Log($"Exception thrown: {e.Message}");
        }

        pingEnabled = false;
    });
}

이에 대응하는 클라이언트 코드는 기본으로만 만들면 됩니다.

using System.Net.Sockets;

namespace ConsoleApp2;

internal class Program
{
    static void Main(string[] args)
    {
        string host = "192.168.0.22";
        int port = 18500;

        Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

        socket.Connect(host, port);
        Log("Connected");

        int received = socket.Receive(new byte[4]);
        Log("Received");

        try
        {
            socket.Receive(new byte[4]);
            Log("Received");
        }
        catch (Exception ex)
        {
            Log($"Exception thrown: {ex.Message}");
        }

        socket.Close();

    }

    private static void Log(string text)
    {
        Console.WriteLine($"[{DateTime.Now:mm ss fff}] {text}");
    }
}

이후, 서버와 클라이언트를 실행하고 나서 클라이언트가 실행된 VM을 Pause 시키면 서버에는 다음과 같은 로그가 남게 됩니다.

[02 05 573] Status changed to TimedOut
[02 36 791] Exception thrown: 연결된 구성원으로부터 응답이 없어 연결하지 못했거나, 호스트로부터 응답이 없어 연결이 끊어졌습니다.

약 31초 만에 끊겼으니 KeepAlive에 설정한 대로 동작했습니다. 테스트를 위해 살짝 parameter를 바꿔보면,

SocketOptionName.KeepAlive == 1
SocketOptionName.TcpKeepAliveInterval == 1
SocketOptionName.TcpKeepAliveRetryCount == 10

이제 실행 결과는 10여 초 만에 끊기도록 나옵니다.

[17 34 240] Status changed to TimedOut
[17 44 580] Exception thrown: 연결된 구성원으로부터 응답이 없어 연결하지 못했거나, 호스트로부터 응답이 없어 연결이 끊어졌습니다.




참고로, Accept 이후 얻게 된 클라이언트 소켓에서 기본값을 재정의하는 것도 가능합니다.

Socket clientSocket = listenSocket.Accept();

clientSocket.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.KeepAlive, true);
clientSocket.SetSocketOption(SocketOptionLevel.Tcp, SocketOptionName.TcpKeepAliveTime, 1);
clientSocket.SetSocketOption(SocketOptionLevel.Tcp, SocketOptionName.TcpKeepAliveInterval, 3);
clientSocket.SetSocketOption(SocketOptionLevel.Tcp, SocketOptionName.TcpKeepAliveRetryCount, 10);

현실적으로는, 저렇게 클라이언트마다 재정의하는 것이 필요한 상황은 거의 없을 것입니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 3/11/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 121  122  123  124  125  126  127  128  129  130  131  [132]  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
1755정성태9/22/201434251오류 유형: 241. Unity Web Player를 설치해도 여전히 설치하라는 화면이 나오는 경우 [4]
1754정성태9/22/201424575VC++: 80. 내 컴퓨터에서 C++ AMP 코드가 실행이 될까요? [1]
1753정성태9/22/201420600오류 유형: 240. Lync로 세미나 참여 시 소리만 들리지 않는 경우 [1]
1752정성태9/21/201441059Windows: 100. 윈도우 8 - RDP 연결을 이용해 VNC처럼 사용자 로그온 화면을 공유하는 방법 [5]
1751정성태9/20/201438917.NET Framework: 464. 프로세스 간 통신 시 소켓 필요 없이 간단하게 Pipe를 열어 통신하는 방법 [1]파일 다운로드1
1750정성태9/20/201423827.NET Framework: 463. PInvoke 호출을 이용한 비동기 파일 작업파일 다운로드1
1749정성태9/20/201423730.NET Framework: 462. 커널 객체를 위한 null DACL 생성 방법파일 다운로드1
1748정성태9/19/201425379개발 환경 구성: 238. [Synergy] 여러 컴퓨터에서 키보드, 마우스 공유
1747정성태9/19/201428390오류 유형: 239. psexec 실행 오류 - The system cannot find the file specified.
1746정성태9/18/201426075.NET Framework: 461. .NET EXE 파일을 닷넷 프레임워크 버전에 상관없이 실행할 수 있을까요? - 두 번째 이야기 [6]파일 다운로드1
1745정성태9/17/201423032개발 환경 구성: 237. 리눅스 Integration Services 버전 업그레이드 하는 방법 [1]
1744정성태9/17/201431035.NET Framework: 460. GetTickCount / GetTickCount64와 0x7FFE0000 주솟값 [4]파일 다운로드1
1743정성태9/16/201420983오류 유형: 238. 설치 오류 - Failed to get size of pseudo bundle
1742정성태8/27/201426947개발 환경 구성: 236. Hyper-V에 설치한 리눅스 VM의 VHD 크기 늘리는 방법 [2]
1741정성태8/26/201421329.NET Framework: 459. GetModuleHandleEx로 알아보는 .NET 메서드의 DLL 모듈 관계파일 다운로드1
1740정성태8/25/201432496.NET Framework: 458. 닷넷 GC가 순환 참조를 해제할 수 있을까요? [2]파일 다운로드1
1739정성태8/24/201426490.NET Framework: 457. 교착상태(Dead-lock) 해결 방법 - Lock Leveling [2]파일 다운로드1
1738정성태8/23/201422040.NET Framework: 456. C# - CAS를 이용한 Lock 래퍼 클래스파일 다운로드1
1737정성태8/20/201419742VS.NET IDE: 93. Visual Studio 2013 동기화 문제
1736정성태8/19/201425565VC++: 79. [부연] CAS Lock 알고리즘은 과연 빠른가? [2]파일 다운로드1
1735정성태8/19/201418149.NET Framework: 455. 닷넷 사용자 정의 예외 클래스의 최소 구현 코드 - 두 번째 이야기
1734정성태8/13/201419805오류 유형: 237. Windows Media Player cannot access the file. The file might be in use, you might not have access to the computer where the file is stored, or your proxy settings might not be correct.
1733정성태8/13/201426330.NET Framework: 454. EmptyWorkingSet Win32 API를 사용하는 C# 예제파일 다운로드1
1732정성태8/13/201434455Windows: 99. INetCache 폴더가 다르게 보이는 이유
1731정성태8/11/201427053개발 환경 구성: 235. 점(.)으로 시작하는 파일명을 탐색기에서 만드는 방법
1730정성태8/11/201422144개발 환경 구성: 234. Royal TS의 터미널(Terminal) 연결에서 한글이 깨지는 현상 해결 방법
... 121  122  123  124  125  126  127  128  129  130  131  [132]  133  134  135  ...