使用C#在LAN上唤醒

时间:2009-05-14 06:42:13

标签: c# wake-on-lan

使用C#实现Wake on LAN的最佳方法是什么?

LAN环境(而非互联网)中的计算机需要此功能。该方法需要足够健壮以处理防火墙和其他此类问题。此外,对于不支持此功能或禁用此功能的系统,还有其他选择吗?

主要目标 - 通过LAN唤醒机器(从关机/休眠状态) - 这是使用C#编程。

请指导。

PS:我遇到过以下情况:

  1. http://blog.memos.cz/index.php/team/2008/06/12/wake-on-lan-in-csharp
  2. http://community.bartdesmet.net/blogs/bart/archive/2006/04/02/3858.aspx
  3. http://www.codeproject.com/KB/IP/cswol.aspx
  4. 然而,我是新手,因此无法确定解决方案是否足够全面。如果有人可以推荐上述任何一篇文章,那会有所帮助。

3 个答案:

答案 0 :(得分:12)

对于WOL问题,您必须澄清三个问题才能使其发挥作用:

  1. 通过以太网电缆发送WOL
  2. 配置您的PC以侦听此类内容 数据包并唤醒
  3. 确保数据包来自 发送者到接收者(防火墙, 网关等。)
  4. 正如您在网上已经发现的那样,现有几种解决方案可以解决C#中编程的第一个问题(在浏览链接之后我将从第一个问题开始)。

    第二个是您只能通过配置网络适配器来实现的。只需打开设备管理器并查看网络适配器的属性(如果存在这样的选项并且您可以启用它)。这是无法编程的,因为每个网络适配器都有该功能的另一个实现以及如何启用它。

    C#也无法解决第三个问题。这是一个纯粹的网络问题,您必须配置路由器,网关,ID系统等,以允许此类数据包并让它从发送方流向接收方。由于WOL数据包始终是广播数据包(dest-ip 255.255.255.255)这一事实,它不会离开您的本地网络,并且将始终从路由器,网关或网络之间的任何其他网桥(例如vpns,等)。

    最后但并非最不重要的是,我只会提醒您,第一个问题可以划分为一些较小的数据包,但据我所知,这些问题都受到您提供的链接的限制。

答案 1 :(得分:3)

我正在尝试 Poul Bak 的回答,但无法唤醒我的目标计算机。在验证第三方应用程序 WakeMeOnLan 实际上能够唤醒我的目标计算机后,我编写了以下对我有用的代码:

void SendWakeOnLan(PhysicalAddress target)
{   
    var header = Enumerable.Repeat(byte.MaxValue, 6);
    var data = Enumerable.Repeat(target.GetAddressBytes(), 16).SelectMany(mac => mac);

    var magicPacket = header.Concat(data).ToArray();
    
    using var client = new UdpClient();

    client.Send(magicPacket, magicPacket.Length, new IPEndPoint(IPAddress.Broadcast, 9));
}

用法:

只需像这样传入目标计算机的mac地址:

SendWakeOnLan(PhysicalAddress.Parse("0A-0B-0C-0D-0E-0F"));

我认为此答案与 Poul Bak 的答案之间的主要区别在于,此代码使用 IPv4 上的广播而不是 IPv4/IPv6 上的多播,而且我的网络设备可能无法正确处理/设置以进行多播。< /p>

答案 2 :(得分:0)

我知道,这个问题很老,但仍然有效。由于在接受的答案中没有看到任何C#,因此我编写了自己的“ Wake On Lan”代码。

我的目标是实现一个通用且简单的Wake On Lan class

  • ipv4 ipv6 双栈一起使用。
  • 可与连接到不同网络的一个或多个网卡(NICS)配合使用。
  • 可使用任何标准十六进制格式的 macaddress 使用。
  • 使用多播(在使用多个NIC的Windows中,广播是错误的)。

使用方法:

您需要的只是您要唤醒的计算机上的有线网卡的 MAC地址。任何标准的十六进制表示都可以。然后像这样调用代码:

string mac = "01-02-03-04-05-06";
await WOL.WakeOnLan(mac);

这是课程:

using System;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.NetworkInformation;
using System.Net.Sockets;
using System.Text.RegularExpressions;
using System.Threading.Tasks;

public static class WOL
{

    public static async Task WakeOnLan(string macAddress)
    {
        byte[] magicPacket = BuildMagicPacket(macAddress);
        foreach (NetworkInterface networkInterface in NetworkInterface.GetAllNetworkInterfaces().Where((n) =>
            n.NetworkInterfaceType != NetworkInterfaceType.Loopback && n.OperationalStatus == OperationalStatus.Up))
        {
            IPInterfaceProperties iPInterfaceProperties = networkInterface.GetIPProperties();
            foreach (MulticastIPAddressInformation multicastIPAddressInformation in iPInterfaceProperties.MulticastAddresses)
            {
                IPAddress multicastIpAddress = multicastIPAddressInformation.Address;
                if (multicastIpAddress.ToString().StartsWith("ff02::1%", StringComparison.OrdinalIgnoreCase)) // Ipv6: All hosts on LAN (with zone index)
                {
                    UnicastIPAddressInformation unicastIPAddressInformation = iPInterfaceProperties.UnicastAddresses.Where((u) =>
                        u.Address.AddressFamily == AddressFamily.InterNetworkV6 && !u.Address.IsIPv6LinkLocal).FirstOrDefault();
                    if (unicastIPAddressInformation != null)
                    {
                        await SendWakeOnLan(unicastIPAddressInformation.Address, multicastIpAddress, magicPacket);
                        break;
                    }
                }
                else if (multicastIpAddress.ToString().Equals("224.0.0.1")) // Ipv4: All hosts on LAN
                {
                    UnicastIPAddressInformation unicastIPAddressInformation = iPInterfaceProperties.UnicastAddresses.Where((u) =>
                        u.Address.AddressFamily == AddressFamily.InterNetwork && !iPInterfaceProperties.GetIPv4Properties().IsAutomaticPrivateAddressingActive).FirstOrDefault();
                    if (unicastIPAddressInformation != null)
                    {
                        await SendWakeOnLan(unicastIPAddressInformation.Address, multicastIpAddress, magicPacket);
                        break;
                    }
                }
            }
        }
    }

    static byte[] BuildMagicPacket(string macAddress) // MacAddress in any standard HEX format
    {
        macAddress = Regex.Replace(macAddress, "[: -]", "");
        byte[] macBytes = new byte[6];
        for (int i = 0; i < 6; i++)
        {
            macBytes[i] = Convert.ToByte(macAddress.Substring(i * 2, 2), 16);
        }

        using (MemoryStream ms = new MemoryStream())
        {
            using (BinaryWriter bw = new BinaryWriter(ms))
            {
                for (int i = 0; i < 6; i++)  //First 6 times 0xff
                {
                    bw.Write((byte)0xff);
                }
                for (int i = 0; i < 16; i++) // then 16 times MacAddress
                {
                    bw.Write(macBytes);
                }
            }
            return ms.ToArray(); // 102 bytes magic packet
        }
    }

    static async Task SendWakeOnLan(IPAddress localIpAddress, IPAddress multicastIpAddress, byte[] magicPacket)
    {
        using (UdpClient client = new UdpClient(new IPEndPoint(localIpAddress, 0)))
        {
            await client.SendAsync(magicPacket, magicPacket.Length, multicastIpAddress.ToString(), 9);
        }
    }
}

工作方式:

该代码通过枚举所有“启动”并连接到您的网络的网卡(通常只是一张网卡)而起作用。它将使用多播向所有连接的网络发送“魔术包”,该多播可与ipv4和ipv6一起使用(不必担心网络泛滥,它只有102个字节)。

要工作,要唤醒的计算机必须具有有线连接(无法唤醒无线计算机,因为它们没有连接到任何网络,因此无法唤醒它们)关闭)。发送数据包的计算机可以无线连接。

防火墙通常没有问题,因为计算机已关闭,因此防火墙处于不活动状态。

您必须确保在计算机的BIOS和网卡中启用了“局域网唤醒”。