Запрос DHCP-сервера в C#

Мне нужно получить сопоставление MAC-адреса и IP-адреса, хранящегося на DHCP-сервере, либо с помощью программы, работающей на самом сервере, либо, предпочтительно, с помощью программы, запущенной на одном из клиентов DHCP.

Я понимаю, что утилита netsh может быть использована для получения дампа, но я не добился большого успеха в этом.

Есть рабочие примеры или намек на это?

У меня есть права администратора на DHCP-сервере

редактировать

Я не хочу использовать arp-кеш, так как это потребует от меня широковещательного пинга (который не разрешен в Windows) или пинга всех возможных IP-адресов подсети (что занимает много времени).

Я уверен, что DHCP-сервер хранит сопоставление MAC-адреса с IP-адресом. Как я могу использовать эту информацию для сопоставления MAC-адреса с IP-адресом?

3 ответа

Для этого вы можете использовать компонент "Объекты DHCP" из Windows 2000 Resource Kit. Несмотря на то, что этот компонент сложно найти, он создан для Windows 2000, по словам Microsoft, в июле 2010 года он не работает, и у него очень мало документации, он работает.

  1. Загрузите Resource Kit Tool с именем " Объекты DHCP", например, здесь, если вы не можете найти его в Microsoft. Это даст вам файл.exe, который, в свою очередь, установит компонент "Объекты DHCP".
  2. Зарегистрировать DHCPOBJS.DLL файл с regsvr32 или создайте приложение COM+ для него. То, что применимо, зависит от того, как COM-компонент будет использоваться в вашей системе.
  3. Используйте Импорт библиотеки типов tlbimp.exe создать управляемую оболочку вокруг DHCPOBJS.DLL теперь, когда он зарегистрирован системой.
  4. В Visual Studio добавьте ссылку на управляемую оболочку. Сгенерированное по умолчанию имя DhcpObjects.dll,

Теперь вы можете написать такой код для компонента:

using DhcpObjects;
class Program {
    static void Main(string[] args) {
        var manager = new Manager();
        var server = dhcpmgr.Servers.Connect("1.2.3.4");
        // query server here
    }
}

Программа установки также предоставляет файл справки Windows, в котором содержится дополнительная документация о том, как запрашивать и управлять сервером DHCP. Раздел "Объектная модель" весьма полезен.

using System;
using System.Runtime.InteropServices;
using System.Collections;
using System.Net;

namespace dhcp
{
// c# class for processed clients

public class dhcpClient
{
    public string hostname { get; set; }
    public string ip       { get; set; }
    public string mac      { get; set; }
}

// structs for use with call to unmanaged code

[StructLayout(LayoutKind.Sequential)]
public struct DHCP_CLIENT_INFO_ARRAY
{
    public uint NumElements;
    public IntPtr Clients;
}

[StructLayout(LayoutKind.Sequential)]
public struct DHCP_CLIENT_UID
{
    public uint DataLength;
    public IntPtr Data;
}

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
public struct DHCP_CLIENT_INFO
{
    public uint ip;
    public uint subnet;

    public DHCP_CLIENT_UID mac;

    [MarshalAs(UnmanagedType.LPWStr)]
    public string ClientName;

    [MarshalAs(UnmanagedType.LPWStr)]
    public string ClientComment;
}

// main

class Program
{
    static void Main()
    {
        try
        {
            // get settings

            String server, subnet;

            Console.Write("Enter server : ");
            server = Console.ReadLine();
            Console.Write("Enter subnet : ");
            subnet = Console.ReadLine();

            // gather clients

            ArrayList clients = findDhcpClients(server, subnet);

            // output results

            Console.WriteLine();

            foreach (dhcpClient d in clients)
                Console.WriteLine(String.Format("{0,-35} {1,-15} {2,-15}", d.hostname, d.ip, d.mac));

            Console.WriteLine('\n' + clients.Count.ToString() + " lease(s) in total");
        }

        catch (Exception e)
        {
            Console.WriteLine(e.ToString());
        }

        Console.ReadLine();
    }

    static ArrayList findDhcpClients(string server, string subnet)
    {
        // set up container for processed clients

        ArrayList foundClients = new ArrayList();

        // make call to unmanaged code

        uint parsedMask     = StringIPAddressToUInt32(subnet);
        uint resumeHandle   = 0;
        uint numClientsRead = 0;
        uint totalClients   = 0;

        IntPtr info_array_ptr;

        uint response = DhcpEnumSubnetClients(
            server,
            parsedMask,
            ref resumeHandle,
            65536,
            out info_array_ptr,
            ref numClientsRead,
            ref totalClients
            );

        // set up client array casted to a DHCP_CLIENT_INFO_ARRAY
        // using the pointer from the response object above

        DHCP_CLIENT_INFO_ARRAY rawClients =
            (DHCP_CLIENT_INFO_ARRAY)Marshal.PtrToStructure(info_array_ptr, typeof(DHCP_CLIENT_INFO_ARRAY));

        // loop through the clients structure inside rawClients 
        // adding to the dchpClient collection

        IntPtr current = rawClients.Clients;

        for (int i = 0; i < (int)rawClients.NumElements; i++)
        {
            // 1. Create machine object using the struct

            DHCP_CLIENT_INFO rawMachine =
                (DHCP_CLIENT_INFO)Marshal.PtrToStructure(Marshal.ReadIntPtr(current), typeof(DHCP_CLIENT_INFO));

            // 2. create new C# dhcpClient object and add to the 
            // collection (for hassle-free use elsewhere!!)

            dhcpClient thisClient = new dhcpClient();

            thisClient.ip = UInt32IPAddressToString(rawMachine.ip);

            thisClient.hostname = rawMachine.ClientName;

            thisClient.mac = String.Format("{0:x2}{1:x2}.{2:x2}{3:x2}.{4:x2}{5:x2}",
                Marshal.ReadByte(rawMachine.mac.Data),
                Marshal.ReadByte(rawMachine.mac.Data, 1),
                Marshal.ReadByte(rawMachine.mac.Data, 2),
                Marshal.ReadByte(rawMachine.mac.Data, 3),
                Marshal.ReadByte(rawMachine.mac.Data, 4),
                Marshal.ReadByte(rawMachine.mac.Data, 5));

            foundClients.Add(thisClient);

            // 3. move pointer to next machine

            current = (IntPtr)((int)current + (int)Marshal.SizeOf(typeof(IntPtr)));
        }

        return foundClients;
    }

    public static uint StringIPAddressToUInt32(string ip)
    {
        // convert string IP to uint IP e.g. "1.2.3.4" -> 16909060

        IPAddress i = System.Net.IPAddress.Parse(ip);
        byte[] ipByteArray = i.GetAddressBytes();

        uint ipUint = (uint)ipByteArray[0] << 24;
        ipUint += (uint)ipByteArray[1] << 16;
        ipUint += (uint)ipByteArray[2] << 8;
        ipUint += (uint)ipByteArray[3];

        return ipUint;
    }

    public static string UInt32IPAddressToString(uint ip)
    {
        // convert uint IP to string IP e.g. 16909060 -> "1.2.3.4"

        IPAddress i = new IPAddress(ip);
        string[] ipArray = i.ToString().Split('.');

        return ipArray[3] + "." + ipArray[2] + "." + ipArray[1] + "." + ipArray[0];
    }

    [DllImport("dhcpsapi.dll", SetLastError = true, CharSet = CharSet.Auto)]
    public static extern uint DhcpEnumSubnetClients(
            string ServerIpAddress,
            uint SubnetAddress,
        ref uint ResumeHandle,
            uint PreferredMaximum,
        out IntPtr ClientInfo,
        ref uint ElementsRead,
        ref uint ElementsTotal
    );
}
}

Будет использовать arp -a сделать трюк... на моей машине я получаю вывод:

У меня есть MAC / IP-адрес, замененный поддельными значениями, чтобы показать результаты...

C:\Documents and Settings\Tom> arp -a

Интерфейс: 10.203.24.196 --- 0xf0007
  Тип физического адреса интернет-адреса
  10.203.24.198         02-50-f3-10-14-06 динамический

C:\Documents and Settings\Tom>

Обстреливая с помощью System.Diagnostics.Process, вы можете перенаправить вывод во входной поток и читать из него...

Надеюсь, это поможет, С наилучшими пожеланиями, Том.

Другие вопросы по тегам