1

在 Windows SDK 中有一个名为computerhardwareids的 CLI 工具

该工具会返回各种 GUID,以便为特定情况选择正确的 HardwareId。

这是在我的 PC 中返回此工具的输出:

Using the BIOS to gather information

Computer Information
--------------------

BIOS Vendor: American Megatrends Inc.
BIOS Version string: 1201
System BIOS Major Release: 4
System BIOS Minor Release: 6

System Manufacturer: To be filled by O.E.M.
System Family: To be filled by O.E.M.
System ProductName: To be filled by O.E.M.
SKU Number: SKU

Enclosure Type: 03 "Desktop"


Hardware IDs
------------
{a8670b03-1d98-5e95-ad4e-c64211eac9df}    <- Manufacturer + Family + ProductName + SKUNumber + BIOS Vendor + BIOS Version + BIOS Major Release + BIOS Minor Release
{01c6b2a2-a2b2-58e4-906d-4677639f1a42}    <- Manufacturer + Family + ProductName + BIOS Vendor + BIOS Version + BIOS Major Release + BIOS Minor Release
{dc5af3fe-c2de-539d-aafd-5061a1634723}    <- Manufacturer + ProductName + BIOS Vendor + BIOS Version + BIOS Major Release + BIOS Minor Release
{d78b474d-dee0-5412-bc9d-e9f7d7783df2}    <- Manufacturer + Family + ProductName + SKUNumber
{7ccbb6f1-9641-5f84-b00d-51ff218a4066}    <- Manufacturer + Family + ProductName
{5a127cba-be28-5d3b-84f0-0e450d266d97}    <- Manufacturer + SKUNumber
{6525c6e5-28e9-5f9c-abe4-20fd82504002}    <- Manufacturer + ProductName
{6525c6e5-28e9-5f9c-abe4-20fd82504002}    <- Manufacturer + Family
{482f3f58-6045-593a-9be4-611717ce4770}    <- Manufacturer + Enclosure Type
{11b4a036-3b64-5421-a372-22c07df10a4d}    <- Manufacturer

我想开发一个通用的使用函数,它应该模仿那个微软工具的功能,返回完全相同的 HardwareIds(完全相同)。

我在 MSDN 上找到了信息,所有输出似乎都记录得很清楚,它包含有关返回此工具的值的信息,但它没有具体说明 WMI 类的属性是什么,它只是说“ Bios ”和“ System ” :

· ComputerHardwareIds 概述

·指定计算机的硬件 ID

我很迷茫,我找不到任何值,例如“Family”、“BIOS Vendor”、“Bios Major Release”、“Bios Minor Release”,我不确定“SKU Number”指的是什么。

我认为这些是 WMI 类,该工具在其中获取所有数据的一部分以制作 guid:

· Win32_BIOS 类

· Win32_BaseBoard 类

· Win32_ComputerSystem 类

· Win32_ComputerSystemProduct 类

请注意,文档也这样说:

然后使用 SHA-1 散列算法将每个字符串转换为 GUID。


这是我试图做的,但我不确定我是否对某些概念或某些价值观有误,它是不完整的,而且我对 Guis 也有问题(用注释行解释):

Private Function GetHardwareId() As Guid

    Dim HardwareId As String = String.Empty

    Dim BIOSVersion, BIOSVendor, BIOSMajorRelease, BIOSMinorRelease,
        SystemManufacturer, SystemFamily, SystemProductName, SKUNumber As String

    ' Get System Info.
    Using wmi As New Management.ManagementObjectSearcher("select * from Win32_ComputerSystem")

        Using SystemInfo As Management.ManagementObject = wmi.Get(0)

            SystemManufacturer = Convert.ToString(SystemInfo.Properties("Manufacturer").Value)
            SystemProductName = Convert.ToString(SystemInfo.Properties("Model").Value)
            SystemFamily = I don't know how to get it.
            SKUNumber = I don't know how to get it.

        End Using

    End Using

    ' Get BIOS Info.
    Using wmi As New Management.ManagementObjectSearcher("select * from Win32_BIOS")

        Using BIOSInfo As Management.ManagementObject = wmi.Get(0)

            BIOSVersion = Convert.ToString(BIOSInfo.Properties("SMBIOSBIOSVersion").Value) 
            BIOSVendor = I don't know how to get it.
            BIOSMajorRelease = I don't know how to get it.
            BIOSMinorRelease = I don't know how to get it.

        End Using

    End Using ' wmi

    HardwareId = BIOSVersion & BIOSVendor & BIOSMajorRelease & BIOSMinorRelease &
                 SystemManufacturer & SystemFamily & SystemProductName & SKUNumber

    ' Here I call other method to encode the resulting string to SHA1 Hash
    HardwareId = ConvertToSHA1(HardwareId)
    ' and then continue below...

    ' But this will not work, 
    ' it throws an exception about missing "-" chars in the SHA1 string.
    ' So Microsoft formats "manualy" the SHA1 string to add some "-"?
    Return Guid.Parse(HardwareId)

End Function
4

3 回答 3

2

我不认为您的问题可以按照您想要的方式解决。但目前也没有理由这样做。

MS 正在从所提供的各种数据的 SHA 哈希中创建确定性 GUID 。如果根据包含 4 个定义的 GUID 命名空间之一的rfc422 标准创建,我们应该能够使用这 4 个命名空间之一从相同的数据重新创建 GUID。

但是,a)我不能和 b)MSDN 的“为计算机指定硬件 ID”状态:the hardware ID for the computer must be produced by the ComputerHardwareIds tool (ComputerHardwareIDs.exe).... 这让我相信他们使用专有方法(Salt、私钥等)或定义自己的命名空间来生成这些。

您的次要/子问题的一些答案:

  • BIOS 版本/发行版 - 根据DTMF.org 规范,表 5 “发行版”位于偏移量 &H14 和 &H15 处,与“版本”不同。但是,它们也可能嵌入在 Win32_BIOS(参见下面的工具)的名称、标题、描述和 SoftwareElementID 属性中。它似乎也被隐藏起来,Win32_BIOS.BiosVersion(1)但它与名称/标题等相同。

我觉得我们的系统相隔几年但具有相同的 Release 值有点奇怪,它可能指的是 SMBios 版本/规范。

  • SKU:根据 MSDN,这是 AKAIdentificationCode

  • 家庭:显然是 BIOS 编码的一部分,但 WMI 没有公开或返回它(还没有?)。

  • 产品名称也是如此,Model您从其他地方获取的值可能只是巧合。

因此,散列中使用的值似乎并没有全部暴露。在我的旧系统上,Family 和 SKU 为空。结果,似乎第一个和第二个ID应该相同,但事实并非如此。

如果 GUID/ID 只能从该工具中获得,我不确定它们的用途或它们对普通应用程序有多大价值。您可能会查看 SDK 的其他部分,以查看是否有程序集等在运行时提供信息。

如果您只是想在下次看到某个系统或设备时对其进行识别,您可以简单地基于 rfc422 编写自己的方法,以确保在您定义的命名空间内具有相同的“非常高的概率”唯一值。像 MS 一样这样做的唯一原因是,如果您要看到来自其他地方的价值,情况并非如此。

最后,我没有费心发布 GUID 制造商,因为它无论如何都不会做你想做的事。


用于获取属性值的 WMI 助手:

Public Sub GetWMIInfo(wmiclass As String)

    Using searcher As New Management.ManagementObjectSearcher("select * from " & wmiclass)

        For Each item As System.Management.ManagementObject In searcher.Get
            DebugProperties(item)
        Next

    End Using
End Sub

' this sub is copied from the watcher answer I gave:
Private Sub DebugProperties(mo As Management.ManagementObject)

    For Each pd As PropertyData In mo.Properties
        If pd.Value IsNot Nothing Then
            ' some props are string arrays, so you can iterate them if you want

            Console.WriteLine("{0} {1}", pd.Name,
                              If(pd.Value IsNot Nothing,
                                 pd.Value.ToString,
                                 "Nothing"))
        End If

    Next
End Sub

输出是这样的:

Caption BIOS Date: XXXXXXXXXXXX Ver: 04.06.04
Description BIOS Date: ##/##/## 11:18:49 Ver: 04.06.04
Manufacturer Dell Inc.
Name BIOS Date: ##/##/## 11:18:49 Ver: 04.06.04
PrimaryBIOS True
ReleaseDate ########000000.000000+000
SerialNumber ######
SMBIOSBIOSVersion A##
SMBIOSMajorVersion #
SMBIOSMinorVersion #
于 2014-08-07T20:09:58.113 回答
1
public ArrayList<String> getWinVendor()
        throws SecurityException, IOException,
        NullPointerException, IndexOutOfBoundsException,
        UnsupportedEncodingException {
    try {
        Process processProduct = Runtime.getRuntime().exec(new String[]{"wmic", "csproduct", "get", "vendor"});
        processProduct.getOutputStream().close();
        BufferedReader output = getOutput(processProduct);
        BufferedReader error = getError(processProduct);
        StringProductList = new ArrayList<String>();
        String line = "", result = "";
        while ((line = output.readLine()) != null) {
            if (!line.toLowerCase().startsWith("vendor") && line.length() > 0) {
                result = getSubStringSubstractEmptyAndTabSpace(line);
                if (result.length() > 0) {
                    StringProductList.add(result);
                } else {
                    StringProductList.add(UNKNOWN);
                }
            }
        }
        if (!StringProductList.isEmpty()) {
            return StringProductList;
        }
    } catch (Exception e) {
        if (e instanceof SecurityException
                || e instanceof IOException
                || e instanceof NullPointerException
                || e instanceof IndexOutOfBoundsException
                || e instanceof UnsupportedEncodingException) {
            e.printStackTrace();
        }
    }
    return null;
}

public BufferedReader getError(Process process) throws SecurityException, IOException,
        NullPointerException, IndexOutOfBoundsException, UnsupportedEncodingException {
    try {
        if (getCmdEncoding() != null) {
            return new BufferedReader(new InputStreamReader(process.getErrorStream(), "UTF-8"));
        }
    } catch (Exception e) {
        if (e instanceof SecurityException
                || e instanceof IOException
                || e instanceof NullPointerException
                || e instanceof IndexOutOfBoundsException) {
            e.printStackTrace();
        }
    }
    return new BufferedReader(new InputStreamReader(process.getErrorStream(), "UTF-8"));
}

public BufferedReader getOutput(Process process) throws SecurityException, IOException,
        NullPointerException, IndexOutOfBoundsException, UnsupportedEncodingException {
    try {
        if (getCmdEncoding() != null) {
            return new BufferedReader(new InputStreamReader(process.getInputStream(), "UTF-8"));
        }
    } catch (Exception e) {
        if (e instanceof SecurityException
                || e instanceof IOException
                || e instanceof NullPointerException
                || e instanceof IndexOutOfBoundsException) {
            e.printStackTrace();
        }
    }
    return new BufferedReader(new InputStreamReader(process.getInputStream(), "UTF-8"));
}

public static String getSubStringSubstractEmptyAndTabSpace(String word)
        throws NullPointerException, IndexOutOfBoundsException {
    if (word.length() > 0) {
        try {
            int length = word.length();
            int start = 0, end = length;
            for (int stringCharacter = 0; stringCharacter < length; stringCharacter++) {
                char c = word.charAt(stringCharacter);
                if (c == ' ' || c == '\t') {
                    start++;
                } else {
                    stringCharacter = length;
                }
            }
            for (int stringCharacter = length - 1; stringCharacter >= 0; stringCharacter--) {
                char c = word.charAt(stringCharacter);
                if (c == ' ' || c == '\t') {
                    end--;
                } else {
                    stringCharacter = -1;
                }
            }
            if (start == length) {
                return "";
            }
            if (end == 0) {
                return "";
            }
            if (start <= length - 1 && end >= 0) {
                return word.substring(start, end);
            }
        } catch (Exception e) {
            if (e instanceof NullPointerException
                    || e instanceof IndexOutOfBoundsException) {
                e.printStackTrace();
            }
        }
    }
    return word;
}
于 2014-11-12T16:44:05.307 回答
1

要生成相同的 GUID,您需要从 SMBIOS 获取值(通常使用 GetSystemFirmwareTable),然后使用“&”字符连接它们。确保字符串使用 UTF-16 编码。然后,您需要使用70ffd812-4c7f-4c7d-0000-000000000000作为命名空间的类型 5 (SHA-1) UUID 生成方案。

于 2017-04-25T19:58:21.473 回答