93 lines
3.6 KiB
C#
93 lines
3.6 KiB
C#
using NetworkResurrector.Abstractions;
|
||
using System.Globalization;
|
||
using System.Net;
|
||
using System.Text.RegularExpressions;
|
||
using System.Threading.Tasks;
|
||
|
||
namespace NetworkResurrector.WakeOnLan
|
||
{
|
||
internal class WolDriver : IWakeOnLanService
|
||
{
|
||
private readonly WolClient _client;
|
||
|
||
public WolDriver(WolClient client)
|
||
{
|
||
_client = client;
|
||
}
|
||
|
||
public async Task<(bool success, string message)> Wake(string macAddress)
|
||
{
|
||
//remove all non 0-9, A-F, a-f characters
|
||
macAddress = Regex.Replace(macAddress, @"[^0-9A-Fa-f]", "");
|
||
|
||
//check if mac adress length is valid
|
||
if (macAddress.Length != 12)
|
||
return (false, "Invalid MAC address.");
|
||
else
|
||
return await Wakeup(macAddress);
|
||
}
|
||
|
||
#region Wakeup
|
||
/// <summary>
|
||
/// Wakes up the machine with the given <paramref name="macAddress"/>.
|
||
/// </summary>
|
||
/// <remarks>
|
||
/// <para>
|
||
/// <note>
|
||
/// <list type="number">
|
||
/// <item> The motherboard must support Wake On LAN.</item>
|
||
/// <item> The NIC must support Wake On LAN.</item>
|
||
/// <item> There must be a wire connecting the motherboard's WOL port to
|
||
/// the NIC's WOL port. Usually there always is a connection on most of
|
||
/// the PCs.</item>
|
||
/// <item> The Wake On LAN feature must be enabled in the motherboard's
|
||
/// BIOS. Usually this is also enabled by default, but you might like to
|
||
/// check again.</item>
|
||
/// <item> The "Good Connection" light on the back of the NIC must be lit
|
||
/// when the machine is off. (By default always good if you are not
|
||
/// facing any network issues)</item>
|
||
/// <item> Port 12287 (0x2FFF) must be open. (By default it should be
|
||
/// open unless some antivirus or any other such program has changed
|
||
/// settings.)</item>
|
||
/// <item> Packets cannot be broadcast across the Internet. That's why
|
||
/// it's called Wake On Lan, not Wake On Internet.</item>
|
||
/// <item> To find your MAC address, run the MSINFO32.EXE tool that is a
|
||
/// part of Windows and navigate to Components > Network > Adapter
|
||
/// or simply type nbtstat -a <your hostname < at command prompt.
|
||
/// e.g. nbtstat -a mymachinename or nbtstat -A 10.2.100.213.
|
||
/// <param name="macAddress">The MAC address of the host which has to be
|
||
/// woken up.</param>
|
||
///
|
||
private async Task<(bool success, string message)> Wakeup(string macAddress)
|
||
{
|
||
_client.Connect(new IPAddress(0xffffffff) /*255.255.255.255 i.e broadcast*/, 0x2fff /*port = 12287*/);
|
||
|
||
if (!_client.SetClientInBrodcastMode())
|
||
return (false, "Remote client could not be set in broadcast mode!");
|
||
|
||
int byteCount = 0;
|
||
byte[] bytes = new byte[102];
|
||
|
||
for (int trailer = 0; trailer < 6; trailer++)
|
||
{
|
||
bytes[byteCount++] = 0xFF;
|
||
}
|
||
|
||
for (int macPackets = 0; macPackets < 16; macPackets++)
|
||
{
|
||
int i = 0;
|
||
for (int macBytes = 0; macBytes < 6; macBytes++)
|
||
{
|
||
bytes[byteCount++] = byte.Parse(macAddress.Substring(i, 2), NumberStyles.HexNumber);
|
||
i += 2;
|
||
}
|
||
}
|
||
|
||
int returnValue = await _client.SendAsync(bytes, byteCount);
|
||
|
||
return (true, $"{returnValue} bytes sent to {macAddress}");
|
||
}
|
||
#endregion
|
||
}
|
||
}
|