使用 JAVA 在 Windows 中为自定义硬件编写 WMI 提供程序

Posted

技术标签:

【中文标题】使用 JAVA 在 Windows 中为自定义硬件编写 WMI 提供程序【英文标题】:Writing a WMI provider for a custom hardware in windows with JAVA 【发布时间】:2019-08-20 06:29:35 【问题描述】:

我有一个连接到 Windows 计算机的自定义硬件设备。我想将此设备的静态信息和动态数据提供给使用 WMI 查询的计算机的其他服务。

根据我的研究,我发现我必须写一个WMI provider。我当前的软件应用程序使用 JPOS 来连接硬件。因此,我必须将 WMI 连接到 Java 应用程序。

我见过C# 和c++ 示例来完成这项任务。我目前的理解是编写一个 C++ wmi 提供程序并使用 JNI 集成到我当前的应用程序中。我已经看到了 JNA 用于query using wmi 的更多示例。然而,我的研究并没有产生任何关于使用 JNA 编写提供程序的有用信息。

编写 C++ 并通过 JNI 集成是处理这种情况的最佳方式吗?或者有没有更好的解决方案?

【问题讨论】:

您目前的理解是正确的,尽管您可能会发现 C# 在提供程序方面更易于使用。提供方与 Java 或 JNA 没有任何关系。它可能是一个 DLL,以某种方式注册。 【参考方案1】:

以下是关于我目前如何为任何想尝试的人解决此问题的一些提示。

1.创建自定义 wmi 类。

Windows wbemtest.exe 工具是您的朋友。这个工具可以挽救你的生命,因为它可以生成新的 wmi 类并对其进行编辑。当以管理员权限打开时,它可以生成自定义 wmi 类,添加属性和修改。

也可以编写一个 .mof 文件来创建自定义类。 .mof 文件的示例如下。

#pragma namespace("\\\\.\\Root\\Cimv2")

class MyTestClass

    [Key] uint32 KeyProperty = 1;
    string Version = "1.1.1";
;

关于运行这个 .mof 文件的信息可以在here找到。

2。添加属性

虽然 webmtester 和 .mof 方法可用于添加属性,但我发现有些 powershell commandlet 很有用。 Stephane van Gulick 的here 是一组强大的 powershell 命令行开关。

3.以编程方式检索属性

以编程方式检索属性的示例 c++ 程序如下。

// PropertyRetrieve.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"
// WMI query to list all properties and values of the root/CIMV2:Detagger class.
// This C++ code was generated using the WMI Code Generator, Version 10.0.13.0
// https://www.robvanderwoude.com/wmigen.php
//
// The generated code was derived from sample code provided by Microsoft:
// https://msdn.microsoft.com/en-us/library/aa390423(v=vs.85).aspx
// The sample code was modified to display multiple properties.
// Most of the original comments from the sample code were left intact.
// Limited testing has been done in Microsoft Visual C++ 2010 Express Edition.

#define _WIN32_DCOM
#include <iostream>
#include <iomanip>
#include <string>
#include <comdef.h>
#include <Wbemidl.h>
using namespace std;

#pragma comment( lib, "wbemuuid.lib" )

HRESULT hr;
IWbemClassObject *pclsObj = NULL;

void DisplayProperty(LPCWSTR propertyname)

    VARIANT vtProperty;
    VariantInit(&vtProperty);
    try
    
        hr = pclsObj->Get(propertyname, 0, &vtProperty, 0, 0);
        if (vtProperty.vt == VT_DISPATCH)
        
            wcout << vtProperty.pdispVal;
        
        else if (vtProperty.vt == VT_BSTR)
        
            wcout << vtProperty.bstrVal;
        
        else if (vtProperty.vt == VT_UI1)
        
            wcout << vtProperty.uiVal;
        
        else if (vtProperty.vt == VT_EMPTY)
        
            wcout << L"[NULL]";
        
    
    catch (...)
    
        wcout.clear();
        wcout << resetiosflags(std::ios::showbase);
    
    VariantClear(&vtProperty);


int main(int argc, char **argv)

    HRESULT hres;

    // Step 1: --------------------------------------------------
    // Initialize COM. ------------------------------------------

    hres = CoInitializeEx(0, COINIT_MULTITHREADED);
    if (FAILED(hres))
    
        cerr << "Failed to initialize COM library. Error code = 0x" << hex << hres << endl;
        return 1; // Program has failed.
    

    // Step 2: --------------------------------------------------
    // Set general COM security levels --------------------------

    hres = CoInitializeSecurity(
        NULL,
        -1,                          // COM authentication
        NULL,                        // Authentication services
        NULL,                        // Reserved
        RPC_C_AUTHN_LEVEL_DEFAULT,   // Default authentication
        RPC_C_IMP_LEVEL_IMPERSONATE, // Default Impersonation
        NULL,                        // Authentication info
        EOAC_NONE,                   // Additional capabilities
        NULL                         // Reserved
    );

    if (FAILED(hres))
    
        cerr << "Failed to initialize security. Error code = 0x" << hex << hres << endl;
        CoUninitialize();
        return 1;                    // Program has failed.
    

    // Step 3: ---------------------------------------------------
    // Obtain the initial locator to WMI -------------------------

    IWbemLocator *pLoc = NULL;

    hres = CoCreateInstance(
        CLSID_WbemLocator,
        0,
        CLSCTX_INPROC_SERVER,
        IID_IWbemLocator, (LPVOID *)&pLoc
    );

    if (FAILED(hres))
    
        cerr << "Failed to create IWbemLocator object. Err code = 0x" << hex << hres << endl;
        CoUninitialize();
        return 1;                 // Program has failed.
    

    // Step 4: -----------------------------------------------------
    // Connect to WMI through the IWbemLocator::ConnectServer method

    IWbemServices *pSvc = NULL;

    // Connect to the root\CIMV2 namespace with
    // the current user and obtain pointer pSvc
    // to make IWbemServices calls.
    hres = pLoc->ConnectServer(
        _bstr_t(L"root\\CIMV2"), // Object path of WMI namespace
        NULL,                      // User name. NULL = current user
        NULL,                      // User password. NULL = current
        0,                         // Locale. NULL indicates current
        NULL,                      // Security flags.
        0,                         // Authority (for example, Kerberos)
        0,                         // Context object
        &pSvc                      // pointer to IWbemServices proxy
    );

    if (FAILED(hres))
    
        cerr << "Could not connect. Error code = 0x" << hex << hres << endl;
        pLoc->Release();
        CoUninitialize();
        return 1;                // Program has failed.
    

    cerr << "Connected to root\\CIMV2 WMI namespace" << endl;
    // Step 5: --------------------------------------------------
    // Set security levels on the proxy -------------------------

    hres = CoSetProxyBlanket(
        pSvc,                        // Indicates the proxy to set
        RPC_C_AUTHN_WINNT,           // RPC_C_AUTHN_xxx
        RPC_C_AUTHZ_NONE,            // RPC_C_AUTHZ_xxx
        NULL,                        // Server principal name
        RPC_C_AUTHN_LEVEL_CALL,      // RPC_C_AUTHN_LEVEL_xxx
        RPC_C_IMP_LEVEL_IMPERSONATE, // RPC_C_IMP_LEVEL_xxx
        NULL,                        // client identity
        EOAC_NONE                    // proxy capabilities
    );

    if (FAILED(hres))
    
        cerr << "Could not set proxy blanket. Error code = 0x" << hex << hres << endl;
        pSvc->Release();
        pLoc->Release();
        CoUninitialize();
        return 1;               // Program has failed.
    

    // Step 6: --------------------------------------------------
    // Use the IWbemServices pointer to make requests of WMI ----

    IEnumWbemClassObject* pEnumerator = NULL;
    hres = pSvc->ExecQuery(
        bstr_t("WQL"),
        bstr_t("SELECT Name,TestValue,Version FROM Detagger"),
        NULL,
        NULL,
        &pEnumerator
    );

    if (FAILED(hres))
    
        cerr << "Query of Detagger class failed. Error code = 0x" << hex << hres << endl;
        pSvc->Release();
        pLoc->Release();
        CoUninitialize();
        return 1;               // Program has failed.
    

    // Step 7: -------------------------------------------------
    // Get the data from the query in step 6 -------------------

    ULONG uReturn = 0;

    while (pEnumerator)
    
        hr = pEnumerator->Next(WBEM_INFINITE, 1, &pclsObj, &uReturn);
        if (hr != 0)
        
            break;
        
        wcout << "Name      : ";
        DisplayProperty((LPCWSTR)L"Name");
        wcout << endl;
        wcout << "TestValue : ";
        DisplayProperty((LPCWSTR)L"TestValue");
        wcout << endl;
        wcout << "Version   : ";
        DisplayProperty((LPCWSTR)L"Version");
        wcout << endl;
        pclsObj->Release();
    

    // Cleanup
    // =======

    pSvc->Release();
    pLoc->Release();
    pEnumerator->Release();
    CoUninitialize();

    cout << "Press anykey to exit.";
    cin.ignore();
    cin.get();

    return 0;   // Program successfully completed.

4.以编程方式修改属性

以编程方式修改属性的示例 C++ 代码如下。此选项需要管理权限。

    _variant_t  var2(L"15");


    IWbemClassObject *detaggerClass = NULL;
    HRESULT dflkj = pSvc->GetObjectW(L"Detagger", 0, NULL, &detaggerClass, NULL);
    IWbemClassObject *detaggerInstance = NULL;
    dflkj = detaggerClass->SpawnInstance(0, &detaggerInstance);

    detaggerInstance->Put(L"TestValue", 0, &var2, CIM_UINT8)
        || Fail("Put failed for 'TestValue'");

    HRESULT er = pSvc->PutInstance(detaggerInstance, WBEM_FLAG_CREATE_OR_UPDATE, NULL, NULL);

这里我修改了 unsigned int 8 变量“TestValue”的值

5.下一步

我的下一个选择是通过 JNA 将 c++ 应用程序连接到主 Java 应用程序。

【讨论】:

以上是关于使用 JAVA 在 Windows 中为自定义硬件编写 WMI 提供程序的主要内容,如果未能解决你的问题,请参考以下文章

如何在 thrift-compiler 中为自定义语言创建自己的 thrift 语言模板?

为自定义视频输入硬件添加对 QuickTime 的支持

Google Analytics 使用 GTag,如何在 JavaScript 中为自定义细分添加代码?

如何在 WooCommerce 中为自定义产品数据选项卡定义图标

如何在 iOS 中为自定义属性设置动画

PSR-4 自动加载如何在 composer 中为自定义库工作?