如何访问 UPnP 设备的服务?

Posted

技术标签:

【中文标题】如何访问 UPnP 设备的服务?【英文标题】:How do I access services of UPnP device? 【发布时间】:2012-11-26 20:19:36 【问题描述】:

设备:Belkin Wemo Switch 开发环境:Windows7 上的 MS VC++ 2010

我正在尝试使用 Windows 中的 C++ 枚举 UPnP 设备的服务。

我有 IUPnPDevice 指针,可以访问多个属性。 我有 IUPnPServices 指针,可以计算正确的服务数量 (7)。 我使用QueryInterface() 获取IEnumVARIANT 指针(似乎成功)。 但是,Next() 方法总是以 HRESULT0x80040500 失败 - 转换为 Windows error 1280 (0x500) - ERROR_ALREADY_FIBER。 这个错误对我没有任何意义。

(我尝试过同时使用 IEnumVARIANTIEnumUnknown - 正如文档中指出的那样,但两者都会产生相同的结果。)

我在下面包含了完整的源文件,以及它产生的输出。 [注意:使用我自己设备的 udn 是硬编码的]

如果有人能提供帮助,我将不胜感激。

最好的问候, 戴夫

代码:

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

#include "stdafx.h"

#include <windows.h>
#include <upnp.h>

static void DumpComError(const TCHAR *api, HRESULT hr);

int _tmain(int argc, _TCHAR* argv[])

  int retcode=-1; // assume failure

  HRESULT hr = CoInitialize(0);
  if (hr==S_OK)
  
    IUPnPDeviceFinder *deviceFinder=0;
    hr = CoCreateInstance(CLSID_UPnPDeviceFinder, 0, CLSCTX_INPROC_SERVER, IID_IUPnPDeviceFinder, (void**)&deviceFinder);
    if (hr==S_OK)
    
      IUPnPDevice *device=0;
      hr = deviceFinder->FindByUDN(L"uuid:Socket-1_0-221239K11002F6", &device);
      if (hr==S_OK)
      
        if (device)
        
          TCHAR *manufacturer=0, *manufacturerUrl=0;
          TCHAR *description=0, *name=0, *modelUrl=0;
          TCHAR *serialNumber=0, *udn=0, *upc=0, *deviceType=0;
          TCHAR *presentationUrl=0;

          device->get_ManufacturerName(&manufacturer);
          device->get_ManufacturerURL(&manufacturerUrl);
          device->get_Description(&description);
          device->get_FriendlyName(&name);
          device->get_ModelURL(&modelUrl);
          device->get_SerialNumber(&serialNumber);
          device->get_UniqueDeviceName(&udn);
          device->get_UPC(&upc);
          device->get_Type(&deviceType);
          device->get_PresentationURL(&presentationUrl);

          _tprintf(_T("MANUFACTURER: %s [%s]\n"), manufacturer, manufacturerUrl);
          _tprintf(_T("MODEL:        %s [%s]\n              [%s]\n"), description, name, modelUrl);
          _tprintf(_T("DEVICE:       serial=%s\n              udn=%s\n              upc=%s\n              type=%s\n"), serialNumber, udn, upc, deviceType);
          _tprintf(_T("URL:          %s\n"), presentationUrl);

          IUPnPServices *services=0;
          hr = device->get_Services(&services);
          if (hr==S_OK)
          
            if (services)
            
              long numberOfServices=0;
              services->get_Count(&numberOfServices);

              if (numberOfServices>0)
              
                IUnknown *unknown=0;
                hr = services->get__NewEnum(&unknown);
                if (hr==S_OK)
                
                  if (unknown)
                  
                    IEnumVARIANT *enumInterface=0;
                    hr = unknown->QueryInterface(IID_IEnumVARIANT,(void**)&enumInterface);
                    if (enumInterface)
                    
                      VARIANT var;
                      unsigned long fetched=0;
                      hr = enumInterface->Next(1, &var, &fetched);

                      if (hr==S_OK)
                      

                      
                      else
                        DumpComError(_T("IEnumVARIANT::Next"), hr);
                    
                    else
                      DumpComError(_T("IUnknown::QueryInterface"), hr);
                  
                  else
                    fprintf(stderr, "Failed to get enumeration interface.\n");
                
                else
                  DumpComError(_T("IUPnPServices::get__NewEnum"), hr);
              
              else
                fprintf(stderr, "No services available.\n");
            
            else
              fprintf(stderr, "Failed to get services collection.\n");
          
          else
            DumpComError(_T("IUPnPDevice::get_Services"), hr);
        
        else
          fprintf(stderr, "Device not found.\n");
      
      else
        DumpComError(_T("IUPnPDeviceFinder::FindByUDN"), hr);
    
    else
      DumpComError(_T("CoCreateIndex"), hr);
  
  else
    DumpComError(_T("CoInitialize"), hr);

  return retcode;


static void AddBoolToString(const TCHAR *name, bool value, TCHAR *buf, int &i, int max)

  if (name && *name && value && buf && i>=0)
    i += _snwprintf_s(&buf[i], max-i, (max-i-1)*sizeof(TCHAR), _T("%s%s=YES"), (i>0? _T("; "): _T("")), name);


static void AddIntToString(const TCHAR *name, int value, TCHAR *buf, int &i, int max)

  if (name && *name && value && buf && i>=0)
    i += _snwprintf_s(&buf[i], max-i, (max-i-1)*sizeof(TCHAR), _T("%s%s=%d"), (i>0? _T("; "): _T("")), name, value);


static void DumpComError(const TCHAR *api, HRESULT hr)

  bool failure   = (hr&0x80000000? true: false);
  bool severe    = (hr&0x40000000? true: false);
  bool microsoft = (hr&0x20000000? false: true);
  bool ntStatus  = (hr&0x10000000? true: false);
  bool xBit      = (hr&0x08000000? true: false);
  int facility   = (hr&0x07FF0000)>>16;
  int code       = (hr&0x0000FFFF);

  TCHAR buf[1024]=0;
  int bufsize = sizeof(buf)/sizeof(TCHAR);
  int i=0;

  AddBoolToString(_T("failure"), failure, buf, i, bufsize);
  AddBoolToString(_T("severe"), severe, buf, i, bufsize);
  AddBoolToString(_T("microsoft"), microsoft, buf, i, bufsize);
  AddBoolToString(_T("ntStatus"), ntStatus, buf, i, bufsize);
  AddBoolToString(_T("xBit"), xBit, buf, i, bufsize);
  AddIntToString(_T("facility"), facility, buf, i, bufsize);
  AddIntToString(_T("code"), code, buf, i, bufsize);

  _ftprintf(stderr, _T("\n%s() failed, hr=0x%08x\n[%s]\n"), api, hr, buf);

输出: 它产生以下输出:

MANUFACTURER: Belkin International Inc. [http://www.belkin.com/]
MODEL:        Belkin Plugin Socket 1.0 [WeMo Switch]
              [http://www.belkin.com/plugin/]
DEVICE:       serial=221239K11002F6
              udn=uuid:Socket-1_0-221239K11002F6
              upc=123456789
              type=urn:Belkin:device:controllee:1
URL:          http://192.168.1.16:49153/pluginpres.html

IEnumVARIANT::Next() failed, hr=0x80040500
[failure=YES; microsoft=YES; facility=4; code=1280]

编辑:

经过许多死胡同,我设法通过手动构建 SOAP 请求并使用 Windows 套接字通过 TCP 发送请求来完成这项工作。棘手的一点是让语法恰到好处,因为我以前没有使用 SOAP 的经验。 [UPnP 有助于识别 IP 地址和端口号 - 因为这些可能会改变]。一旦启动并运行 - 它实际上比 UPnP 界面简单得多。如果您有兴趣,请告诉我,我可以发布代码……它并不能直接回答我在这里提出的问题,所以用这个细节来回答我的问题是没有意义的。

但是,如果您有兴趣,请告诉我,我可以发布代码。

干杯, 戴夫

【问题讨论】:

注意:从这个页面 [issackelly.com/blog/2012/07/30/wemo-hacking/] 你可以看到服务列表,所以理论上我应该可以直接使用 IUPnPServices::get_Item() 访问服务但是我不能找出 serviceId 的正确格式(第一个参数)... 注2:我尝试使用 IUPnPService::get_Item(L"urn:Belkin:serviceId:basicevent1", &serviceId);但是返回的 HRESULT 与之前相同(0x80040500) 是的 @user390935 ,发布您的代码。我正在寻找平台中立的 SSDP 客户端。 【参考方案1】:

0x80040500 的 HRESULT 不是你想的那样,而是UPNP_E_INVALID_DOCUMENT。有关如何产生这种歧义的解释,请参阅my answer in another SO question。

我的猜测是您的 Belkin 设备提供了不符合要求的设备描述或服务描述 XML。不符合并不一定意味着格式不正确,UPnP 规范有大量的次要要求。尝试 Intel Developer Tools 中的 Device Spy(另一个答案底部的链接),如果设备弹出,则从其上的同一套件运行 Device Validator。

【讨论】:

谢谢你...我应该知道 Facility 是 4。我猜这意味着 Windows 不喜欢设备返回的响应。 是的,这是我的猜测。 HRESULT 是与 OS/2 兼容 20 年的可怕遗产。所有“用户定义的代码”只有一个 FACILITY_ITF,这本身就已经够糟糕的了,但微软正在为自己的新 API 使用相同的工具,它本可以使用另一个未使用的代码。【参考方案2】:

我的经历是相似的,因为 UPnPDeviceFinder 根本不起作用。它从不发送 UPnP 搜索数据包,因此设备不响应​​。让它工作的唯一方法是,如果您还使用 Windows 媒体播放器或“Cast To Device”菜单(即 WMP)来启动搜索。因为它是 UPnPDeviceFinder 将返回一些设备,只有当它们恰好正在广播时,但即使找到 XBox(另一个 Microsoft 产品)在没有其他活动的示例中也不起作用。

【讨论】:

以上是关于如何访问 UPnP 设备的服务?的主要内容,如果未能解决你的问题,请参考以下文章

.Net的UPnP库[关闭]

Android UPnP/DLNA 应用以及cling 2.1.1依赖库

UpnP Hacking

群晖 媒体服务器

Win10哪些服务可以关闭

获取 UPnP 设备的 IP 地址