Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)
(시리즈 글이 3개 있습니다.)
닷넷: 2183. C# - eFriend Expert OCX 예제를 .NET Core/5+ Console App에서 사용하는 방법
; https://www.sysnet.pe.kr/2/0/13482

닷넷: 2186. 한국투자증권 KIS Developers OpenAPI의 C# 래퍼 버전 - eFriendOpenAPI NuGet 패키지
; https://www.sysnet.pe.kr/2/0/13498

닷넷: 2199. C# - 한국투자증권 KIS Developers OpenAPI의 WebSocket Ping, Pong 처리
; https://www.sysnet.pe.kr/2/0/13521




C# - 한국투자증권 KIS Developers OpenAPI의 WebSocket Ping, Pong 처리

C#으로 KIS Developers OpenAPIWebSocket을 연결해 두면,

_webSocket = new ClientWebSocket();
await _webSocket.ConnectAsync(new Uri(url), connectTimeout.Token);

이후 ReceiveAsync 호출 시 대략 30초 후에 이런 예외가 (내부적으로) 발생하면서 연결이 끊깁니다.

System.Net.WebSockets.WebSocketException
  HResult=0x80004005
  Message=The remote party closed the WebSocket connection without completing the close handshake.
  Source=System.Net.WebSockets
  StackTrace:
   at System.Net.WebSockets.ManagedWebSocket.ThrowEOFUnexpected() in /_/src/libraries/System.Net.WebSockets/src/System/Net/WebSockets/ManagedWebSocket.cs:line 1445

그래서 이후 WebSocket에 대한 Send/Receive 호출을 하게 되면 이렇게 오류가 발생합니다.

System.Net.WebSockets.WebSocketException
  HResult=0x80004005
  Message=The WebSocket is in an invalid state ('Aborted') for this operation. Valid states are: 'Open, CloseReceived'
  Source=System.Net.WebSockets
  StackTrace:
   at System.Net.WebSockets.WebSocketValidate.ThrowIfInvalidState(WebSocketState currentState, Boolean isDisposed, WebSocketState[] validStates) in /_/src/libraries/Common/src/System/Net/WebSockets/WebSocketValidate.cs:line 54
   at System.Net.WebSockets.ManagedWebSocket.SendAsync(ReadOnlyMemory`1 buffer, WebSocketMessageType messageType, WebSocketMessageFlags messageFlags, CancellationToken cancellationToken) in /_/src/libraries/System.Net.WebSockets/src/System/Net/WebSockets/ManagedWebSocket.cs:line 287
--- End of stack trace from previous location ---
   at eFriendOpenAPI.eFriendClient.<국내주식실시간체결가>d__52.MoveNext() in C:\KIS\eFriendOpenAPI\eFriendClient.websocket.cs:line 44
   at ConsoleApp1.Program.<Main>d__0.MoveNext() in C:\KIS\ConsoleApp1\Program.cs:line 111
   at ConsoleApp1.Program.<Main>(String[] args)

결론부터 말하면, 위의 오류를 해결하기 위해서는 KeepAliveInterval을 0으로 설정해야 (결과적으로 꺼야) 합니다.

_webSocket = new ClientWebSocket();
_webSocket.Options.KeepAliveInterval = TimeSpan.FromSeconds(0);

그러니까, (문서상으로는 2분이라고 되어 있는데) KeepAliveInterval의 기본값이 30초라서 저런 현상이 발생했던 것입니다.

일례로, KeepAliveInterval을 더 낮추면,

_webSocket.Options.KeepAliveInterval = TimeSpan.FromSeconds(5); // 5초

더 빠르게 System.Net.WebSockets.WebSocketException 예외가 발생합니다.




지난 글에 따라,

C# - ClientWebSocket의 Ping, Pong 처리
; https://www.sysnet.pe.kr/2/0/13518

KeepAliveInterval을 0으로 설정하면 ClientWebSocket은 더 이상 Pong 신호를 전송하지 않게 되고 위에서 언급한 웹소켓 연결 끊김 현상도 발생하지 않게 됩니다.

그런데 이상하지 않나요? 상대방이 살아 있는지 확인하기 위해 Ping/Pong을 사용하는 것인데, 오히려 그 Ping/Pong을 하면 연결이 끊기는 아이러니한 상황이 발생한 것입니다.

혹시, ClientWebSocket 측의 Ping이 아닌 Pong을 보내는 방식 때문에 그런 걸까요? ^^ 사실 아래의 글은 그것을 증명하기 위해 썼던 것입니다.

C# - Reflection을 이용한 ClientWebSocket의 Ping 호출
; https://www.sysnet.pe.kr/2/0/13519

결과는, (Pong 대신) Ping으로 해도 웹소켓은 끊깁니다. 이 같은 상황을 정리해 보면, "아마도" KIS Developers OpenAPI 측의 ping/pong 수신 코드 내에 오류가 있는 것이 아닌가... 생각됩니다. 즉, Ping/Pong 처리 시에 예외가 발생해서 웹소켓 자원이 그냥 해제되는 것입니다. 실제로 C# 오류 메시지에는 "The remote party closed the WebSocket connection without completing the close handshake."라고 "close handshake" 없이 일방적으로 끊었다는 메시지가 나옵니다.

그나마 이 와중에 다행인 점은, Ping/Pong을 하지 않아도 연결 개체를 서버 측에서 끊지 않는다는 것입니다. ^^;




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

[연관 글]






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

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

비밀번호

댓글 작성자
 




... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13178정성태12/1/20225223Windows: 215. Win32 API 금지된 함수 - IsBadXxxPtr 유의 함수들이 안전하지 않은 이유파일 다운로드1
13177정성태11/30/20225925오류 유형: 829. uwsgi 설치 시 fatal error: Python.h: No such file or directory
13176정성태11/29/20224762오류 유형: 828. gunicorn - ModuleNotFoundError: No module named 'flask'
13175정성태11/29/20226631오류 유형: 827. Python - ImportError: cannot import name 'html5lib' from 'pip._vendor'
13174정성태11/28/20225044.NET Framework: 2073. C# - VMMap처럼 스택 메모리의 reserve/guard/commit 상태 출력파일 다운로드1
13173정성태11/27/20225824.NET Framework: 2072. 닷넷 응용 프로그램의 스레드 스택 크기 변경
13172정성태11/25/20225514.NET Framework: 2071. 닷넷에서 ESP/RSP 레지스터 값을 구하는 방법파일 다운로드1
13171정성태11/25/20225181Windows: 214. 윈도우 - 스레드 스택의 "red zone"
13170정성태11/24/20225430Windows: 213. 윈도우 - 싱글 스레드는 컨텍스트 스위칭이 없을까요?
13169정성태11/23/20226024Windows: 212. 윈도우의 Protected Process (Light) 보안 [1]파일 다운로드2
13168정성태11/22/20225371제니퍼 .NET: 31. 제니퍼 닷넷 적용 사례 (9) - DB 서비스에 부하가 걸렸다?!
13167정성태11/21/20225396.NET Framework: 2070. .NET 7 - Console.ReadKey와 리눅스의 터미널 타입
13166정성태11/20/20225124개발 환경 구성: 651. Windows 사용자 경험으로 WSL 환경에 dotnet 런타임/SDK 설치 방법
13165정성태11/18/20224952개발 환경 구성: 650. Azure - "scm" 프로세스와 엮인 서비스 모음
13164정성태11/18/20225945개발 환경 구성: 649. Azure - 비주얼 스튜디오를 이용한 AppService 원격 디버그 방법
13163정성태11/17/20225861개발 환경 구성: 648. 비주얼 스튜디오에서 안드로이드 기기 인식하는 방법
13162정성태11/15/20226920.NET Framework: 2069. .NET 7 - AOT(ahead-of-time) 컴파일
13161정성태11/14/20226230.NET Framework: 2068. C# - PublishSingleFile로 배포한 이미지의 역어셈블 가능 여부 (난독화 필요성) [4]
13160정성태11/11/20226096.NET Framework: 2067. C# - PublishSingleFile 적용 시 native/managed 모듈 통합 옵션
13159정성태11/10/20229413.NET Framework: 2066. C# - PublishSingleFile과 관련된 옵션 [3]
13158정성태11/9/20225499오류 유형: 826. Workload definition 'wasm-tools' in manifest 'microsoft.net.workload.mono.toolchain' [...] conflicts with manifest 'microsoft.net.workload.mono.toolchain.net7'
13157정성태11/8/20226162.NET Framework: 2065. C# - Mutex의 비동기 버전파일 다운로드1
13156정성태11/7/20227118.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점파일 다운로드1
13155정성태11/4/20226591디버깅 기술: 183. TCP 동시 접속 (연결이 아닌) 시도를 1개로 제한한 서버
13154정성태11/3/20226090.NET Framework: 2063. .NET 5+부터 지원되는 GC.GetGCMemoryInfo파일 다운로드1
13153정성태11/2/20227381.NET Framework: 2062. C# - 코드로 재현하는 소켓 상태(SYN_SENT, SYN_RECV)
... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...