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

... 76  77  78  79  80  81  82  83  84  85  [86]  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
11786정성태11/29/201818622오류 유형: 505. OpenGL.NET 예제 실행 시 "Managed Debugging Assistant 'CallbackOnCollectedDelegate'" 예외 발생
11785정성태11/21/201820969디버깅 기술: 120. windbg 분석 사례 - ODP.NET 사용 시 Finalizer에서 System.AccessViolationException 예외 발생으로 인한 비정상 종료
11784정성태11/18/201820233Graphics: 32. .NET으로 구현하는 OpenGL (7), (8) - Matrices and Uniform Variables, Model, View & Projection Matrices파일 다운로드1
11783정성태11/18/201818340오류 유형: 504. 윈도우 환경에서 docker가 설치된 컴퓨터 간의 ping IP 주소 풀이 오류
11782정성태11/18/201817452Windows: 152. 윈도우 10에서 사라진 "Adapters and Bindings" 네트워크 우선순위 조정 기능 - 두 번째 이야기
11781정성태11/17/201820825개발 환경 구성: 422. SFML.NET 라이브러리 설정 방법 [1]파일 다운로드1
11780정성태11/17/201821849오류 유형: 503. vcpkg install bzip2 빌드 에러 - "Error: Building package bzip2:x86-windows failed with: BUILD_FAILED"
11779정성태11/17/201822738개발 환경 구성: 421. vcpkg 업데이트 [1]
11778정성태11/14/201820064.NET Framework: 803. UWP 앱에서 한 컴퓨터(localhost, 127.0.0.1) 내에서의 소켓 연결
11777정성태11/13/201820526오류 유형: 502. Your project does not reference "..." framework. Add a reference to "..." in the "TargetFrameworks" property of your project file and then re-run NuGet restore.
11776정성태11/13/201818546.NET Framework: 802. Windows에 로그인한 계정이 마이크로소프트의 계정인지, 로컬 계정인지 알아내는 방법
11775정성태11/13/201820364Graphics: 31. .NET으로 구현하는 OpenGL (6) - Texturing파일 다운로드1
11774정성태11/8/201818776Graphics: 30. .NET으로 구현하는 OpenGL (4), (5) - Shader파일 다운로드1
11773정성태11/7/201818460Graphics: 29. .NET으로 구현하는 OpenGL (3) - Index Buffer파일 다운로드1
11772정성태11/6/201820406Graphics: 28. .NET으로 구현하는 OpenGL (2) - VAO, VBO파일 다운로드1
11771정성태11/5/201819438사물인터넷: 56. Audio Jack 커넥터의 IR 적외선 송신기 - 두 번째 이야기 [1]
11770정성태11/5/201827819Graphics: 27. .NET으로 구현하는 OpenGL (1) - OpenGL.Net 라이브러리 [3]파일 다운로드1
11769정성태11/5/201818814오류 유형: 501. 프로젝트 msbuild Publish 후 connectionStrings의 문자열이 $(ReplacableToken_...)로 바뀌는 문제
11768정성태11/2/201819271.NET Framework: 801. SOIL(Simple OpenGL Image Library) - Native DLL 및 .NET DLL 제공
11767정성태11/1/201820181사물인터넷: 55. New NodeMcu v3(ESP8266)의 IR LED (적외선 송신) 제어파일 다운로드1
11766정성태10/31/201822259사물인터넷: 54. 아두이노 환경에서의 JSON 파서(ArduinoJson) 사용법
11765정성태10/26/201819172개발 환경 구성: 420. Visual Studio Code - Arduino Board Manager를 이용한 사용자 정의 보드 선택
11764정성태10/26/201824068개발 환경 구성: 419. MIT 라이선스로 무료 공개된 Detours API 후킹 라이브러리 [2]
11763정성태10/25/201821015사물인터넷: 53. New NodeMcu v3(ESP8266)의 https 통신
11762정성태10/25/201821446사물인터넷: 52. New NodeMCU v3(ESP8266)의 http 통신파일 다운로드1
11761정성태10/25/201821402Graphics: 26. 임의 축을 기반으로 3D 벡터 회전파일 다운로드1
... 76  77  78  79  80  81  82  83  84  85  [86]  87  88  89  90  ...