Microsoft MVP성태의 닷넷 이야기
.NET Framework: 942. C# - WOL(Wake On Lan) 구현 [링크 복사], [링크+제목 복사],
조회: 21559
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

C# - WOL(Wake On Lan) 구현

예전에 구현했던,

라즈베리 파이를 이용해 원격 컴퓨터의 전원 스위치 제어
; https://www.sysnet.pe.kr/2/0/11726

Synology NAS(DS216+II)에 FTDI 장치 연결 후 C#(.NET Core)으로 DTR 제어
; https://www.sysnet.pe.kr/2/0/11734

전원 스위치를 이용해 원격으로 잘 제어하던 PC를 교체했더니,

2020년 작업 PC ^^
; https://www.sysnet.pe.kr/0/0/522

다시 저 피복 벗기고 하는 식의 작업을 새 PC에 하는 것이 귀찮아졌습니다. 그래도 꽤 괜찮게 써먹었던 기능이라서 없으면 무척 아쉬울 것 같아 차선책으로 WoL(Wake on Lan) 기능으로 넘어갔는데요, 검색해 보면 아래의 글이 꽤 적절하게 잘 설명하고 있기 때문에,

Wake On Lan(WOL) : 원격으로 컴퓨터 켜기 설정 및 사용
; https://neoray.org/281

자세한 것은 넘어가고, 곧바로 소스 코드 구현을 보겠습니다. 검색해 보면 아주 많은 소스 코드를 볼 수 있지만, 사실 프로토콜 자체가 매우 쉬우므로 아래와 같이 간단하게 구현할 수 있습니다.

using System;
using System.Collections.Generic;
using System.Globalization;
using System.IO;
using System.Net;
using System.Net.NetworkInformation;
using System.Net.Sockets;
using System.Text.RegularExpressions;

namespace wol
{
    class Program
    {
        const int WOL_PACKET_LEN = 102;

        // Wake-on-Lan (WoL) in C#
        // https://www.fluxbytes.com/csharp/wake-lan-wol-c/
        static void Main(string[] args)
        {
            byte[] wolBuffer = GetWolPacket(args[0]);

            UdpClient udp = new UdpClient();
            udp.EnableBroadcast = true;

            IPAddress ipAddress = IPAddress.Parse("255.255.255.255");
            udp.Send(wolBuffer, wolBuffer.Length, ipAddress.ToString(), 7);
            udp.Send(wolBuffer, wolBuffer.Length, ipAddress.ToString(), 9);
        }

        private static byte[] GetWolPacket(string macAddress)
        {
            byte[] datagram = new byte[WOL_PACKET_LEN];

            byte[] macBuffer = StringToBytes(macAddress);

            MemoryStream ms = new MemoryStream(datagram);
            BinaryWriter bw = new BinaryWriter(ms);

            // 6바이트의 0xff를 선두에 채우고,
            for (int i = 0; i < 6; i++)
            {
                bw.Write((byte)0xff);
            }

            // 이후 WoL로 깨울 PC가 소유한 Network Adapter의 MAC 주소를 16번 반복
            for (int i = 0; i < 16; i++)
            {
                bw.Write(macBuffer, 0, macBuffer.Length);
            }

            return datagram;
        }

        private static byte[] StringToBytes(string macAddress)
        {
            macAddress = Regex.Replace(macAddress, "[-|:]", ""); // Remove any semicolons or minus characters present in our MAC address
            byte[] buffer = new byte[macAddress.Length / 2];

            for (int i = 0; i < macAddress.Length; i += 2)
            {
                string digit = macAddress.Substring(i, 2);
                buffer[i / 2] = byte.Parse(digit, NumberStyles.HexNumber);
            }

            return buffer;
        }
    }
}

여기서 재미있는 것은 Port인데요, 여러 소스 코드들을 보면 0번, 3번, 7번, 9번 등을 사용해 다소 혼란스러울 수 있는데 아래의 문서를 보면,

Wake-on-LAN
; https://en.wikipedia.org/wiki/Wake-on-LAN#Magic_packet

although it is typically sent as a UDP datagram to port 0, 7 or 9, or directly over Ethernet as EtherType 0x0842.

The internet with local broadcasting - some routers permit a packet received from the internet to be broadcast to the entire LAN [26]; the default TCP or UDP ports preconfigured to relay WOL requests are usually ports 7 (Echo Protocol) and/or 9 (Discard Protocol). This proxy setting must be enabled in the router, and port forwarding rules may need to be configured in its embedded firewall in order to accept magic packets coming from the internet side to these restricted port numbers, and to allow rebroadcasting them on the local network (normally to the same ports and the same TCP or UDP protocol). Such routers may also be configurable to use different port numbers for this proxying service.


Router의 영향을 고려해 7번 또는 9번을 사용하는 것이 좋아 보입니다. (이 글의 소스 코드에서는 2개 모두 사용했습니다.) 또한, 255.255.255.255 자체의 Broadcasting에 대한 제약을 따져 봤을 때,

UDP 브로드캐스트 주소 255.255.255.255와 192.168.0.255의 차이점과 이를 고려한 C# UDP 서버/클라이언트 예제
; https://www.sysnet.pe.kr/2/0/11368

모든 어댑터를 통해 보낼 수 있도록 Send 부분을 다음과 같이 변경해 주면 더 좋을 것입니다.

foreach (IPAddress ipAddress in GetDirectedBroadcastAddresses())
{
    udp.Send(wolBuffer, wolBuffer.Length, ipAddress.ToString(), port);
}

private static IPAddress[] GetDirectedBroadcastAddresses()
{
    List<IPAddress> list = new List<IPAddress>();

    foreach (NetworkInterface item in NetworkInterface.GetAllNetworkInterfaces())
    {
        if (item.NetworkInterfaceType == NetworkInterfaceType.Loopback)
        {
            continue;
        }

        if (item.OperationalStatus != OperationalStatus.Up)
        {
            continue;
        }

        UnicastIPAddressInformationCollection unicasts = item.GetIPProperties().UnicastAddresses;

        foreach (UnicastIPAddressInformation unicast in unicasts)
        {
            IPAddress ipAddress = unicast.Address;

            if (ipAddress.AddressFamily != AddressFamily.InterNetwork)
            {
                continue;
            }

            byte[] addressBytes = ipAddress.GetAddressBytes();
            byte[] subnetBytes = unicast.IPv4Mask.GetAddressBytes();

            if (addressBytes.Length != subnetBytes.Length)
            {
                continue;
            }

            byte[] broadcastAddress = new byte[addressBytes.Length];
            for (int i = 0; i < broadcastAddress.Length; i++)
            {
                broadcastAddress[i] = (byte)(addressBytes[i] | (subnetBytes[i] ^ 255));
            }

            list.Add(new IPAddress(broadcastAddress));
        }
    }

    return list.ToArray();
}

Github에도 소스 코드 및 빌드된 바이너리를 올려두었습니다.

stjeong / Utilities / wol
; https://github.com/stjeong/Utilities/tree/master/wol

Utilities.zip - wol.exe
; https://github.com/stjeong/Utilities/releases/

참고로, 컴퓨터를 끄는 것은 "작업 스케줄러"에 "shutdown /h" 명령어를 원하는 시간에 실행하도록 등록하면 됩니다.




직접 실습을 해보니까, 제가 테스트한 4대의 컴퓨터 중에 구형 컴퓨터 2대(각각 11년, 8년)는 BIOS 설정과 윈도우의 Adapter 속성창에서 WoL 관련 설정을 했지만 동작하지 않았습니다. 반면 4년 된 컴퓨터와 새로 구매한 컴퓨터는 정상 동작했습니다. (그러고 보니, 저 4대의 컴퓨터가 모두 ASUS 보드군요. ^^)




검색하다가 낚인 것이 있어 ^^ 공유해 봅니다. 아래의 글에 대한 제목만 보면 Hyper-V의 가상 머신(VM)을 WoL 방식으로 깨우는 호스트 측의 기능이 있을 것 같은데요,

Wake on LAN for Hyper-V Guests
; https://deploymentpros.wordpress.com/2016/11/28/wake-on-lan-for-hyper-v-guests/

PowerShell Script (psHyper-V_WoL.ps1)
; https://gallery.technet.microsoft.com/scriptcenter/Wake-on-LAN-for-Hyper-V-21578819

실상은 Hyper-V 호스트가 VM에 대한 WoL을 지원하는 것은 아닙니다. 단지, 위의 스크립트가 UDP 소켓을 만들어 Receive로 대기하고 있다가 VM이 소유한 네트워크 어댑터의 WoL 신호가 들어오면 Start-VM 명령어를 이용하는 역할을 하는 것입니다.




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

[연관 글]






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

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)
1780정성태10/15/201424154오류 유형: 249. The application-specific permission settings do not grant Local Activation permission for the COM Server application with CLSID
1779정성태10/15/201419663오류 유형: 248. Active Directory에서 OU가 지워지지 않는 경우
1778정성태10/10/201418103오류 유형: 247. The Netlogon service could not create server share C:\Windows\SYSVOL\sysvol\[도메인명]\SCRIPTS.
1777정성태10/10/201421207오류 유형: 246. The processing of Group Policy failed. Windows attempted to read the file \\[도메인]\sysvol\[도메인]\Policies\{...GUID...}\gpt.ini
1776정성태10/10/201418226오류 유형: 245. 이벤트 로그 - Name resolution for the name _ldap._tcp.dc._msdcs.[도메인명]. timed out after none of the configured DNS servers responded.
1775정성태10/9/201419374오류 유형: 244. Visual Studio 디버깅 (2) - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
1774정성태10/9/201426566개발 환경 구성: 246. IIS 작업자 프로세스의 20분 자동 재생(Recycle)을 끄는 방법
1773정성태10/8/201429750.NET Framework: 471. 웹 브라우저로 다운로드가 되는 파일을 왜 C# 코드로 하면 안되는 걸까요? [1]
1772정성태10/3/201418522.NET Framework: 470. C# 3.0의 기본 인자(default parameter)가 .NET 1.1/2.0에서도 실행될까? [3]
1771정성태10/2/201428038개발 환경 구성: 245. 실행된 프로세스(EXE)의 명령행 인자를 확인하고 싶다면 - Sysmon [4]
1770정성태10/2/201421665개발 환경 구성: 244. 매크로 정의를 이용해 파일 하나로 C++과 C#에서 공유하는 방법 [1]파일 다운로드1
1769정성태10/1/201424082개발 환경 구성: 243. Scala 개발 환경 구성(JVM, 닷넷) [1]
1768정성태10/1/201419497개발 환경 구성: 242. 배치 파일에서 Thread.Sleep 효과를 주는 방법 [5]
1767정성태10/1/201424609VS.NET IDE: 94. Visual Studio 2012/2013에서의 매크로 구현 - Visual Commander [2]
1766정성태10/1/201422424개발 환경 구성: 241. 책 "프로그래밍 클로저: Lisp"을 읽고 나서. [1]
1765정성태9/30/201425996.NET Framework: 469. Unity3d에서 transform을 변수에 할당해 사용하는 특별한 이유가 있을까요?
1764정성태9/30/201422256오류 유형: 243. 파일 삭제가 안 되는 경우 - The action can't be comleted because the file is open in System
1763정성태9/30/201423843.NET Framework: 468. PDB 파일을 연동해 소스 코드 라인 정보를 알아내는 방법파일 다운로드1
1762정성태9/30/201424538.NET Framework: 467. 닷넷에서 EIP/RIP 레지스터 값을 구하는 방법 [1]파일 다운로드1
1761정성태9/29/201421541.NET Framework: 466. 윈도우 운영체제의 보안 그룹 이름 및 설명 문자열을 바꾸는 방법파일 다운로드1
1760정성태9/28/201419804.NET Framework: 465. ICorProfilerInfo::GetILToNativeMapping 메서드가 0x80131358을 반환하는 경우
1759정성태9/27/201430963개발 환경 구성: 240. Visual C++ / x64 환경에서 inline-assembly를 매크로 어셈블리로 대체하는 방법파일 다운로드1
1758정성태9/23/201437812개발 환경 구성: 239. 원격 데스크톱 접속(RDP)을 기존의 콘솔 모드처럼 사용하는 방법 [1]
1757정성태9/23/201418379오류 유형: 242. Lync로 모임 참여 시 소리만 들리지 않는 경우 - 두 번째 이야기
1756정성태9/23/201427374기타: 48. NVidia 제품의 과다한 디스크 사용 [2]
1755정성태9/22/201434175오류 유형: 241. Unity Web Player를 설치해도 여전히 설치하라는 화면이 나오는 경우 [4]
... 121  122  123  124  125  126  127  128  129  130  [131]  132  133  134  135  ...