访问 iOS 设备和设备元数据

设备 API 可通过适用于 iOS 的 Home API 进行访问。 将以下软件包导入到您的应用中:

import GoogleHomeSDK
import GoogleHomeTypes

如需了解详情,请参阅 iOS 上的数据模型

错误处理

Home API 中的某些方法会抛出 HomeError,因此我们建议您使用 do-catch 块来捕获这些调用中的 HomeError

处理 HomeError 时,请检查其 codemessage 字段,了解问题所在。

任何未处理的错误都会导致应用崩溃。

如需了解详情,请参阅错误处理

如需查看示例,请参阅向设备发送命令

示例调用

获取设备列表

通过对 Home 对象的引用,调用 devices() 以获取可访问设备的 Query。调用 Querybatched() 方法,该方法会在每次设备元数据发生更改时发出一个反映 Home 当前状态的 Set。或者,调用 Query.list() 可获取可用设备的快照。这是一种便捷方法,用于订阅 batched() 流并返回第一个已发出值。Query.stream() 会生成一个流,该流会在设备元数据(例如名称、房间或结构)发生变化时发出新值。在内部,此函数使用 batched(),并且仅发出已更改的属性。

// Get a list of all devices accessible to the user
let homeDevices = try await self.home.devices().list()

这样,您就可以访问每个设备的状态,并向设备发送支持的命令。

获取设备的类型

如需获取与设备关联的设备类型,请读取设备的 types 属性,该属性会返回 DeviceTypeController

调用 DeviceTypeController.subscribe(_:) 可订阅特定设备类型的更新:

let devices = try await self.home.devices().list()
if let device = devices.first(where: { $0.id == myDeviceId }) {
  var receivedUpdate1 = false
  var receivedUpdate2 = false
  device.types.subscribe(OnOffLightDeviceType.self)
    .assertNoFailure()
    .sink { device in
      if !receivedUpdate1 {
        receivedUpdate1 = true
        Task {
          try await device.matterTraits.onOffTrait?.on()
        }
        return
      }
      if !receivedUpdate2 {
        receivedUpdate2 = true
        return
      }
      fatalError("Received unexpected update")
    }
}

如果设备不支持指定的设备类型,则会返回立即完成的 Empty Publisher

如果设备支持特定设备类型,您可以通过调用 get() 获取该类型的句柄:

if let device = devices.first(where: { $0.id == myDeviceId }) {
  let deviceType = await device.types.get(OnOffLightDeviceType.self)
}

如果设备不支持指定的类型,则返回 nil

调用 DeviceTypeController.subscribeAll() 以获取 DeviceTypeCollectionPublisher。此类可让您检查设备是否具有特定设备类型:

if let device = devices.first(where: { $0.id == myDeviceId }) {
  device.types.subscribeAll()
    .assertNoFailure()
    .sink { types in
      let lightDeviceType = types[OnOffLightDeviceType.self]
      let fanDeviceType = types[FanDeviceType.self]
    }
}

获取设备类型特征

设备类型是读取特征的入口点,因为它们会将设备分解为各个功能部分(如 Matter 中的端点)。

它们还会考虑特征冲突,以防设备具有两种设备类型,而这两种设备类型可能具有相同的特征。例如,如果设备既是音箱又是可调光灯,则它将具有两个“开启/关闭”特征和两个“调光控制”特征。

当设备具有两个名称相同的特征时,可能会发生另一种特征冲突。例如,onOff 可以指标准 OnOff 特征的实例,也可以指制造商定义的 OnOff 特征的实例。为消除可能存在的有关预期特征的任何歧义,请通过每种设备类型上的两个特征集合之一来引用特征。

对于标准特征(即与 Matter 标准集群类似的特征),请使用 matterTraits。例如,如需获取“可调光灯”设备类型的特定特征,请执行以下操作:

if let dimmableLightDeviceType =
  await device.types.get(DimmableLightDeviceType.self)
{
  // Accessing standard trait on the type.
  let levelControlTrait =
    dimmableLightDeviceType.matterTraits.levelControlTrait.self
}

对于 Google 特征,请使用 googleTraits

if let doorbellDeviceType = await device.types.get(GoogleDoorbellDeviceType.self) {
  // Accessing Google trait on the type.
  let doorbellPressTrait =
    doorbellDeviceType.googleTraits.doorbellPressTrait.self
}

如需访问制造商特定的特征,请通过 traits 属性引用该特征,但要在前面加上制造商的软件包名称:

let deviceType = await device1?.types.get(OnOffLightDeviceType.self)
// Accessing custom trait on the type.
if let spinnerTrait = deviceType?.traits[ExampleOrganization.SpinnerTrait.self] {
  let rpmVal = spinnerTrait.attributes.rpm
}

读取设备状态

请看以下示例,了解如何从设备的 On/Off 特征中检查 OnOff 属性:

let lightDevices = devices.filter {
  $0.types.contains(OnOffLightDeviceType.self)
}
let light1 = lightDevices.first
let lightDeviceTypeOptional = await light1?.types.get(OnOffLightDeviceType.self)
if let onOffTrait = lightDeviceTypeOptional?.matterTraits.onOffTrait {
  let onOffVal = onOffTrait.attributes.onOff
}

获取具有特定特征的设备列表

如需获取具有特定特征的设备列表,您需要遍历设备、每个设备的设备类型以及每个设备类型的特征。例如,如需获取住宅中所有具有“开/关”特征的设备的列表,请执行以下操作:

// Get all light devices that support levelControl
var levelControlDevices: [HomeDevice] = []
var allDevices = try await home.devices().list()
for device in allDevices {
  if let deviceType = await device.types.get(OnOffLightDeviceType.self) {
    if deviceType.traits.contains(Matter.LevelControlTrait.self) {
      levelControlDevices.append(device)
    }
  }
}

如需查看 Home API 中可用的特征的完整列表,请参阅 iOS 上的特征索引

获取具有类似设备类型的设备列表

如需获取代表住宅中所有灯具的设备列表,请执行以下操作:

// Get a list of devices with similar device types (lights)
let lightDevices =
  try await self.home.devices().list().compactMap {
    $0.types.contains(DimmableLightDeviceType.self)
      || $0.types.contains(OnOffLightDeviceType.self)
      || $0.types.contains(ColorTemperatureLightDeviceType.self)
      || $0.types.contains(ExtendedColorLightDeviceType.self)
  }

Home API 中有多种设备类型可以代表核心设备类型。例如,没有“灯”设备类型。相反,有四种不同的设备类型可以表示灯,如上例所示。因此,若要全面了解住宅中的更高级别设备类型,必须包含多种设备类型。

如需查看 Home API 中提供的设备类型及其特征的完整列表,请参阅 iOS 上支持的设备类型

获取设备的供应商名称、供应商 ID 或产品 ID

BasicInformationTrait 特征包含供应商 ID、产品 ID、产品名称和设备的序列号等信息:

guard
  let vendorName =
    basicInfoTrait.attributes.vendorName
else {
  fatalError("Failed to get vendorName")
}
guard
  let vendorID =
    basicInfoTrait.attributes.vendorID
else {
  fatalError("Failed to get vendorID")
}
guard
  let productID =
    basicInfoTrait.attributes.productID
else {
  fatalError("Failed to get productID")
}

面向设备制造商的云到云设备识别

如果您是设备制造商并构建 Cloud-to-cloud 设备,为了通过 BasicInformation 特征识别 Cloud-to-cloud 设备,您可以在其 SYNC 响应中添加以下字符串字段:

  • 连接标准联盟 (CSA) 签发的供应商 ID: "matterOriginalVendorId": "0xfff1",

  • 唯一标识供应商产品的商品标识符: "matterOriginalProductId": "0x1234",

  • 设备的唯一标识符,以制造商特定的方式构建: "matterUniqueId": "matter-device-id",

输入这些字符串字段时,请使用您的Matter供应商 ID 和产品 ID(如果有)。如果您不是 CSA 会员,并且未获分配这些 ID,则可以将 matterOriginalVendorIdmatterOriginalProductId 字段留空,并提供 matterUniqueId 作为标识符。

以下示例 SYNC 响应展示了这些字段的用法:

{
  "requestId": "ff36a3cc-ec34-11e6-b1a0-64510650abcf",
  "payload": {
    "agentUserId": "1836.15267389",
    "devices": [
      {
        "id": "456",
        "type": "action.devices.types.LIGHT",
        "traits": [
          "action.devices.traits.OnOff",
          "action.devices.traits.Brightness",
          "action.devices.traits.ColorSetting",
        ],
        "willReportState": true,
        "deviceInfo": { ... },
        "matterOriginalVendorId": "0xfff1",
        "matterOriginalProductId": "0x1234",
        "matterUniqueId": "matter-device-id",
        "otherDeviceIds": [
          {
            "deviceId": "local-device-id",
          }
        ]
      }
    ]
  }
}

如需了解详情,请参阅 Cloud-to-cloud SYNC 文档

设备和特征元数据

Home API 中的设备和特征具有相关联的元数据,这些元数据有助于管理应用中的用户体验。

Home API 中的每个特征都包含一个 sourceConnectivity 属性,其中包含有关特征在线状态和位置(本地或远程路由)的信息。

获取设备的主要类型

某些设备可能会通过 Home API 显示多种设备类型。 为了确保用户在应用中看到适合其设备的选项(例如设备控制和建议的自动化操作),最好检查设备类型是否为设备的主要类型。

if let deviceType =
  await device?.types.get(HumiditySensorDeviceType.self)
{
  if deviceType.metadata.isPrimaryType {
    print("Humidity Sensor is the primary type on this device.")
  } else {
    print("Humidity Sensor isn't the primary type on this device.")
  }
}

检查特征是否处于在线状态

读取 connectivityState 属性以检查特征的连接性:

let levelControlConnectivity =
  levelControlTrait.metadata.sourceConnectivity
  .connectivityState

如果设备未连接到互联网,某些特征(通常是 Google smart home 特征)可能会显示为离线。这是因为这些特征基于云,没有本地路由。

检查设备的连接

设备的连接性实际上是在设备类型级别进行检查的,因为某些设备支持多种设备类型。返回的状态是相应设备上所有特征的连接状态的组合。

let lightConnectivity =
  dimmableLightDeviceType.metadata.sourceConnectivity
  .connectivityState

如果设备类型混杂且没有互联网连接,则可能会出现 partiallyOnline 状态。Matter 标准特征可能仍处于在线状态(由于本地路由),但基于云的特征将处于离线状态。

检查特征的网络路由

特征的语言区域也可在 Home API 中使用。dataSourceLocality 表示特征是通过远程(通过云)、本地(通过本地 Hub)还是对等互连(直接从设备到设备,无需 Hub)进行路由的。

例如,当应用正在启动且尚未连接到用于设备连接的 hub 或服务器时,可能会出现未知位置信息值 unspecified。这些设备无法访问,并且会因命令或事件而无法响应互动请求。客户端可以自行决定如何处理此类设备。

let levelControlLocality =
  levelControlTrait.metadata.sourceConnectivity
  .dataSourceLocality

检查设备的网络路由

与连接性类似,本地性是在设备类型级别进行检查的。返回的状态是相应设备上所有特征的本地化状态的组合。

let lightLocality =
  dimmableLightDeviceType.metadata.sourceConnectivity.dataSourceLocality

在与 partiallyOnline 连接类似的情况下,可能会出现 mixed 状态:某些特征基于云,而另一些特征基于本地。

更改设备的名称

调用 setName(_:) 方法可更改设备的名称:

let updatedDevice = try await theDevice.setName("new device name")

更改设备的名称时,原始 HomeDevice 结构体保持不变,更改反映在返回的更新后的 HomeDevice 对象中。

如果名称超过 60 个 Unicode 码位(字符)的限制,系统会截断名称,但不会抛出任何错误。开发者负责处理过长的名称,例如,可以决定是否要告知用户名称将被截断。

API 列表

创建 Home 的实例后,可以通过该实例访问以下设备 API:

API 说明
device(id:) 返回指定设备的 Publisher,该 Publisher 会在设备状态发生变化时发出设备状态。
devices() 获取 Google 账号中所有结构中的所有设备。返回一个 Query<HomeDevice>,其中包含更多检索和过滤选项。

获得 HomeDevice 后,您可以通过它访问以下 API:

API 说明
id 设备的唯一系统 ID。
name 设备的用户提供名称。
structureID 设备所分配到的结构的 ID。返回 String?
roomID 设备所分配到的房间的 ID。返回 String?
types 获取设备上的特定类型或所有可用类型。
isMatterDevice 如果设备由 Matter 提供支持。
sourceConnectivity 设备的来源连接,表示设备特征的聚合连接状态和网络位置。