Microsoft MVP성태의 닷넷 이야기
.NET Framework: 1057. C# - CoAP 서버 및 클라이언트 제작 (UDP 소켓 통신) [링크 복사], [링크+제목 복사],
조회: 20455
글쓴 사람
정성태 (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]

... 121  [122]  123  124  125  126  127  128  129  130  131  132  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
10873정성태11/25/201525469.NET Framework: 540. C# - 부동 소수 계산 왜 이렇게 나오죠? (2) [3]파일 다운로드1
10872정성태11/24/201532439.NET Framework: 539. C# - 부동 소수 계산 왜 이렇게 나오죠? (1) [1]
10871정성태11/23/201528059오류 유형: 313. SignTool Error: No certificates were found that met all the given criteria.
10870정성태11/23/201528923오류 유형: 312. 윈도우 10 TH2 (버전 1511) 업데이트가 안되는 경우 [1]
10869정성태11/23/201524688오류 유형: 311. certutil 실행 오류 - 0x80070057 [1]
10868정성태11/20/201524564제니퍼 .NET: 25. 제니퍼 닷넷 적용 사례 (5) - RestSharp 라이브러리의 CPU High 현상파일 다운로드1
10867정성태10/18/201527128.NET Framework: 538. Thread.Abort로 인해 프로세스가 종료되는 현상
10866정성태10/14/201523339.NET Framework: 537. C# - Reflection의 박싱 없이 값 형식을 다루는 방법파일 다운로드1
10865정성태10/13/201523117.NET Framework: 536. Thread.Abort의 스레드 종료 지연파일 다운로드1
10864정성태10/12/201521322.NET Framework: 535. aspnet.config 파일의 설정을 읽는 방법
10863정성태10/9/201526134.NET Framework: 534. ASP.NET 응용 프로그램이 예외로 프로세스가 종료된다면?
10862정성태10/9/201524486오류 유형: 310. 비주얼 스튜디오 - Unspecified error (Exception from HRESULT: 0x80004005 (E_FAIL))
10861정성태10/9/201529143기타: 54. 도서: 시작하세요! C# 6.0 프로그래밍: 기본 문법부터 실전 예제까지 (2)
10860정성태10/5/201526595개발 환경 구성: 277. IIS AppPool의 시작/중단에 대한 이벤트 로그 확인 방법
10859정성태10/5/201527787.NET Framework: 533. C#에서 string 형식이 primitive일까요? [6]
10858정성태10/2/201524288VS.NET IDE: 105. Visual Studio의 단위 테스트 작성 시 Fakes를 이용한 메서드 재정의 방법 [1]파일 다운로드1
10857정성태10/1/201520321VS.NET IDE: 104. Visual C++ 프로젝트의 빌드 이벤트에서 환경 변수 사용하는 방법
10856정성태9/30/201531582.NET Framework: 532. WPF DataGrid의 데이터 바인딩 시 리플렉션의 부하는 어느 정도일까요?파일 다운로드1
10855정성태9/30/201521298.NET Framework: 531. C# - XSLT 내의 javascript에 전달한 XML 노드의 타입은?
10854정성태9/30/201521810오류 유형: 309. C# - 포인터를 쓰는 경우 VerificationException이 발생한다면?
10853정성태9/21/201519474오류 유형: 308. 공백 문자를 포함한 계정명의 권한으로 Visual Studio 확장을 설치할 때 오류 발생
10852정성태9/17/201524523VC++: 92. C++ 생성자를 DLL로부터 동적 로드해 객체를 생성한다면? [2]파일 다운로드1
10851정성태9/15/201524271.NET Framework: 530. C# - 중위식을 후위식으로 변환하는 예제파일 다운로드1
10850정성태9/14/201522946.NET Framework: 529. C# - volatile 키워드로 인한 차이점을 발생시키는 예제 [1]파일 다운로드1
10849정성태9/14/201557048오류 유형: 307. CLR20r3 오류 해결을 위해 mscorlib.dll을 덮어쓸때 주의할 점 [12]
10848정성태9/8/201527419VS.NET IDE: 103. Visual Studio의 Ctrl + F5 실행 동작파일 다운로드1
... 121  [122]  123  124  125  126  127  128  129  130  131  132  133  134  135  ...