Microsoft MVP성태의 닷넷 이야기
.NET Framework: 1057. C# - CoAP 서버 및 클라이언트 제작 (UDP 소켓 통신) [링크 복사], [링크+제목 복사],
조회: 11973
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)

C# - CoAP 서버 및 클라이언트 제작 (UDP 소켓 통신)

HTTP 프로토콜에서 TLS 및 모바일 망의 효율을 위해 UDP를 적용하게 된 것처럼,

.NET 5에서의 네트워크 라이브러리 개선 (2) - HTTP/2, HTTP/3 관련
; https://www.sysnet.pe.kr/2/0/12504#http3

IoT 세계의 저전력을 목표로 하는 기기에서 TCP는 부담스러울 수밖에 없습니다. ^^ 그래서 나온 것이, UDP에 기반을 둔 CoAP(The Constrained Application Protocol)입니다.

비록, IoT 기기를 위한 프로토콜이긴 하지만 사실 그냥 일반적인 UDP 통신 래퍼로 여겨도 좋을 수준인데요, 그런 의미에서 ^^ C# 콘솔 프로그램으로 간단하게 CoAP 서버 및 클라이언트를 만들어 보겠습니다.

당연하겠지만, 이미 NuGet에 패키지가 있습니다. ^^

smeshlink/CoAP.NET
; https://github.com/smeshlink/CoAP.NET

CoAP.NET
; https://www.nuget.org/packages/CoAP/
// Install-Package CoAP -Version 1.1.0

CoAP.NET.Core
; https://www.nuget.org/packages/CoAP.NET.Core/
// Install-Package CoAP.NET.Core -Version 1.1.0

일단, 서버는 CoAP.NET의 README.md에서 설명한 대로 다음과 같이 간단하게 제작할 수 있습니다.

using CoAP.Server;
using System;
using System.Diagnostics;

class HelloWorldResource : Resource
{
    public HelloWorldResource()
        : base("hello-world")
    {
        Attributes.Title = "GET a friendly greeting!";
    }

    protected override void DoGet(CoapExchange exchange)
    {
        Console.WriteLine($"temperature data: {exchange.Request.PayloadString}");
        exchange.Respond("Hello World from CoAP.NET!");
    }
}

// Install-Package CoAP -Version 1.1.0
class Server
{
    static void Main(String[] args)
    {
        CoAP.CoapConfig cc = new CoAP.CoapConfig();
        cc.MaxRetransmit = 2;

        CoapServer server = new CoapServer(cc, 60100);
        server.Add(new HelloWorldResource());
        server.Start();

        Console.WriteLine("Press any key to exit...");
        Console.ReadLine();
    }
}

/* 실행 결과
DEBUG - Starting CoAP server
DEBUG - BlockwiseLayer uses MaxMessageSize: 1024 and DefaultBlockSize:512
DEBUG - Starting endpoint bound to [::]:60100
Press any key to exit...
*/

HelloWorldResource 타입의 base 생성자에 "hello-world"를 전달하고 있는데 ASP.NET Web API라면 일종의 "[Route("hello-world")]" 설정이라고 보면 됩니다.

어찌 보면, IoC 컨테이너 없는 단순한 수준의 Web API 템플릿이라고 봐도 무방할 정도입니다.




이렇게 만든 CoAP 서버를 소비하는 클라이언트 역시 간단하게 만들 수 있습니다.

// CoAP.Client - A CoAP Example Client
// https://github.com/smeshlink/CoAP.NET/tree/master/CoAP.Example/CoAP.Client

using CoAP;
using System;
using System.Diagnostics;

// Install-Package CoAP -Version 1.1.0
class Program
{
    static void Main(string[] args)
    {
        Request request = new Request(Method.GET);
        request.URI = new Uri("coap://127.0.0.1:60100/hello-world");
        request.SetPayload("0", MediaType.TextPlain); // 간단한 데이터 전송
        request.Send();

        Response response = request.WaitForResponse();
        if (response == null)
        {
            Console.WriteLine("Failed to get response");
        }
        
    }
}

/* 출력 결과: 서버
temperature data: 0
*/

/* 출력 결과: 클라이언트
DEBUG - BlockwiseLayer uses MaxMessageSize: 1024 and DefaultBlockSize:512
DEBUG - Starting endpoint bound to [::]:61379
DEBUG - Scheduling retransmission for CON-GET ID=-1, Token=21773E8A, Options=[URI-Port=60100, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Send request, failed transmissions: 0
DEBUG - Stored open request by KeyID[59156 for ], KeyToken[21-77-3E-8A]
DEBUG - Exchange got response: Cleaning up KeyID[59156 for ]
DEBUG - Cancel retransmission for -->
DEBUG - CON-GET ID=59156, Token=21773E8A, Options=[URI-Port=60100, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Exchange completed: Cleaning up KeyToken[21-77-3E-8A]
*/

보는 바와 같이 UDP 통신이지만, Request/Response 기반으로 동작해 마치 HTTP REST API 호출하듯이 사용할 수 있습니다. 사용법이 너무 간단하죠. ^^ 결국, 일반 UDP 통신을 구현하는 몇몇 상황에서도 CoAP를 이용하는 것이 더 좋은 선택이 될 수 있을 정도입니다.

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




경우에 따라 IoT 기기에서 데이터를 서버가 받아줄 때까지 무한정 재시도를 하기는 힘들 것입니다. 이를 위해 Send 전에 MaxRetransmit을 설정하면,

...[생략]...
// 재현을 위해 일부러 없는 포트로 전송
request.URI = new Uri("coap://127.0.0.1:13290/hello-world");
request.MaxRetransmit = 2;
request.Send();

다음과 같이 0, 1, 2까지 총 3번의 요청을 전송합니다.

DEBUG - BlockwiseLayer uses MaxMessageSize: 1024 and DefaultBlockSize:512
DEBUG - Starting endpoint bound to [::]:49943
DEBUG - Scheduling retransmission for CON-GET ID=-1, Token=026840C6, Options=[URI-Port=190, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Send request, failed transmissions: 0
DEBUG - Stored open request by KeyID[53977 for ], KeyToken[02-68-40-C6]
DEBUG - Timeout: retransmit message, failed: 1, message: CON-GET ID=53977, Token=026840C6, Options=[URI-Port=190, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Scheduling retransmission for CON-GET ID=53977, Token=026840C6, Options=[URI-Port=190, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Send request, failed transmissions: 1
DEBUG - Stored open request by KeyID[53977 for ], KeyToken[02-68-40-C6]
DEBUG - Timeout: retransmit message, failed: 2, message: CON-GET ID=53977, Token=026840C6, Options=[URI-Port=190, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Scheduling retransmission for CON-GET ID=53977, Token=026840C6, Options=[URI-Port=190, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Send request, failed transmissions: 2
DEBUG - Stored open request by KeyID[53977 for ], KeyToken[02-68-40-C6]
DEBUG - Start Mark-And-Sweep with 0 entries
DEBUG - Timeout: retransmission limit reached, exchange failed, message: CON-GET ID=53977, Token=026840C6, Options=[URI-Port=190, URI-Path=hello-world, Content-Type=text/plain], "0"
DEBUG - Exchange completed: Cleaning up KeyToken[02-68-40-C6]
DEBUG - Exchange completed: Cleaning up KeyToken[02-68-40-C6]
DEBUG - Exchange completed: Cleaning up KeyToken[02-68-40-C6]
Press any key to continue . . .

혹은 Send 후 Response를 받는 메서드에서 timeout을 설정하는 것도 가능합니다.

Request request = new Request(Method.GET);
request.URI = new Uri("coap://127.0.0.1:190/hello-world");
request.SetPayload("0", MediaType.TextPlain);
request.MaxRetransmit = 2;
request.Send();

Response response = request.WaitForResponse(1000);

그럼, MaxRetransmit 설정 횟수에 상관없이 무조건 1000ms 지난 후에 WaitForResponse는 null을 반환하게 됩니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 5/5/2021]

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

비밀번호

댓글 작성자
 



2022-09-22 02시52분
[LGCoap] 회사에서 Iot 장비를 개발 중에
테스트 Coap 서버를 구축해야 했습니다.
공유해주신 자료 덕분에 쉽게 구축하여 테스트 했습니다^^
좋은 정보 감사합니다!!^^
[guest]

... [61]  62  63  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12120정성태1/19/202011152.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
12119정성태1/17/202011192디버깅 기술: 160. Windbg 확장 DLL 만들기 (3) - C#으로 만드는 방법
12118정성태1/17/202011869개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기 [1]
12117정성태1/15/202010856디버깅 기술: 159. C# - 디버깅 중인 프로세스를 강제로 다른 디버거에서 연결하는 방법파일 다운로드1
12116정성태1/15/202011329디버깅 기술: 158. Visual Studio로 디버깅 시 sos.dll 확장 명령어를 (비롯한 windbg의 다양한 기능을) 수행하는 방법
12115정성태1/14/202011085디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법파일 다운로드1
12114정성태1/13/202012950디버깅 기술: 156. C# - PDB 파일로부터 심벌(Symbol) 및 타입(Type) 정보 열거 [1]파일 다운로드3
12113정성태1/12/202013577오류 유형: 590. Visual C++ 빌드 오류 - fatal error LNK1104: cannot open file 'atls.lib' [1]
12112정성태1/12/202010113오류 유형: 589. PowerShell - 원격 Invoke-Command 실행 시 "WinRM cannot complete the operation" 오류 발생
12111정성태1/12/202013410디버깅 기술: 155. C# - KernelMemoryIO 드라이버를 이용해 실행 프로그램을 숨기는 방법(DKOM: Direct Kernel Object Modification) [16]파일 다운로드1
12110정성태1/11/202011992디버깅 기술: 154. Patch Guard로 인해 블루 스크린(BSOD)가 발생하는 사례 [5]파일 다운로드1
12109정성태1/10/20209891오류 유형: 588. Driver 프로젝트 빌드 오류 - Inf2Cat error -2: "Inf2Cat, signability test failed."
12108정성태1/10/20209965오류 유형: 587. Kernel Driver 시작 시 127(The specified procedure could not be found.) 오류 메시지 발생
12107정성태1/10/202010889.NET Framework: 877. C# - 프로세스의 모든 핸들을 열람 - 두 번째 이야기
12106정성태1/8/202012385VC++: 136. C++ - OSR Driver Loader와 같은 Legacy 커널 드라이버 설치 프로그램 제작 [1]
12105정성태1/8/202010975디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법
12104정성태1/7/202011645DDK: 9. 커널 메모리를 읽고 쓰는 NT Legacy driver와 C# 클라이언트 프로그램 [4]
12103정성태1/7/202014412DDK: 8. Visual Studio 2019 + WDK Legacy Driver 제작- Hello World 예제 [1]파일 다운로드2
12102정성태1/6/202011974디버깅 기술: 152. User 권한(Ring 3)의 프로그램에서 _ETHREAD 주소(및 커널 메모리를 읽을 수 있다면 _EPROCESS 주소) 구하는 방법
12101정성태1/5/202011314.NET Framework: 876. C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
12100정성태1/3/20209333.NET Framework: 875. .NET 3.5 이하에서 IntPtr.Add 사용
12099정성태1/3/202011661디버깅 기술: 151. Windows 10 - Process Explorer로 확인한 Handle 정보를 windbg에서 조회 [1]
12098정성태1/2/202011250.NET Framework: 874. C# - 커널 구조체의 Offset 값을 하드 코딩하지 않고 사용하는 방법 [3]
12097정성태1/2/20209803디버깅 기술: 150. windbg - Wow64, x86, x64에서의 커널 구조체(예: TEB) 구조체 확인
12096정성태12/30/201911772디버깅 기술: 149. C# - DbgEng.dll을 이용한 간단한 디버거 제작 [1]
12095정성태12/27/201913217VC++: 135. C++ - string_view의 동작 방식
... [61]  62  63  64  65  66  67  68  69  70  71  72  73  74  75  ...