Implementa la app de entrega local

Para admitir la entrega local, debes compilar una app que controle estos intents de casa inteligente:

  • IDENTIFY: Admite el descubrimiento de dispositivos inteligentes que se pueden controlar de forma local. El controlador de intents extrae los datos que muestra tu dispositivo inteligente durante el descubrimiento y los envía en una respuesta a Google.
  • EXECUTE: Admite la ejecución de comandos.
  • QUERY: Admite la consulta del estado del dispositivo.
  • REACHABLE_DEVICES: Admite el descubrimiento de dispositivos finales que se pueden controlar de forma local detrás de un dispositivo concentrador (o puente) (opcional).

Esta app se ejecuta en los dispositivos Google Home o Google Nest del usuario y conecta tu dispositivo inteligente a Asistente. Puedes crear la app con TypeScript (opción preferida) o JavaScript.

Te recomendamos TypeScript, ya que puedes aprovechar las vinculaciones para asegurarte de forma estática que los datos que muestra tu app coincidan con los tipos que espera la plataforma.

Para obtener más detalles sobre la API, consulta la referencia de la API del SDK de Local Home.

En los siguientes fragmentos, se muestra cómo puedes inicializar la app de entrega local y conectar tus controladores.

Independiente
import App = smarthome.App;
const localHomeApp: App = new App("1.0.0");
localHomeApp
  .onIdentify(identifyHandler)
  .onExecute(executeHandler)
  .listen()
  .then(() => {
    console.log("Ready");
  });
Concentrador
import App = smarthome.App;
const localHomeApp: App = new App("1.0.0");
localHomeApp
  .onIdentify(identifyHandler)
  .onReachableDevices(reachableDevicesHandler)
  .onExecute(executeHandler)
  .listen()
  .then(() => {
    console.log("Ready");
  });

Crear tu proyecto

Si quieres implementar tu app de entrega local, debes compilar un paquete de JavaScript para tu código y todas sus dependencias.

Usa el inicializador de proyectos de la app de entrega local para iniciar la estructura de proyecto adecuada con la configuración del agrupador que prefieras.

Plantillas de proyectos

Para seleccionar la configuración del agrupador, ejecuta el comando npm init como se muestra en los siguientes ejemplos:

Ninguno

TypeScript sin configuración de agrupador:

npm init @google/local-home-app project-directory/ --bundler none

Estructura del proyecto:

project-directory/
├── node_modules/
├── package.json
├── .gitignore
├── index.ts
├── test.ts
├── tsconfig.json
├── tslint.json
└── serve.js

Reemplaza project-directory por un directorio nuevo que contendrá el proyecto de la app de entrega local.

Webpack

TypeScript con configuración del agrupador webpack:

npm init @google/local-home-app project-directory/ --bundler webpack

Estructura del proyecto:

project-directory/
├── node_modules/
├── package.json
├── .gitignore
├── index.ts
├── test.ts
├── tsconfig.json
├── tslint.json
├── webpack.config.web.js
├── webpack.config.node.js
└── serve.js

Reemplaza project-directory por un directorio nuevo que contendrá el proyecto de la app de entrega local.

Resumen

TypeScript con la configuración del agrupador Rollup:

npm init @google/local-home-app project-directory/ --bundler rollup

Estructura del proyecto:

project-directory/
├── node_modules/
├── package.json
├── .gitignore
├── index.ts
├── test.ts
├── tsconfig.json
├── tslint.json
├── rollup.config.js
└── serve.js

Reemplaza project-directory por un directorio nuevo que contendrá el proyecto de la app de entrega local.

Paquete

TypeScript con configuración del agrupador Parcel:

npm init @google/local-home-app project-directory/ --bundler parcel

Estructura del proyecto:

project-directory/
├── node_modules/
├── package.json
├── .gitignore
├── index.ts
├── test.ts
├── tsconfig.json
├── tslint.json
└── serve.js

Reemplaza project-directory por un directorio nuevo que contendrá el proyecto de la app de entrega local.

Realiza tareas comunes a nivel de proyecto

El proyecto generado admite las siguientes secuencias de comandos de npm:

Paquete
cd project-directory/
npm run build

Esta secuencia de comandos compila código fuente de TypeScript y empaqueta tu app con sus dependencias para el entorno de ejecución de Chrome en el subdirectorio dist/web y el entorno de ejecución Node.js en el subdirectorio dist/node.

Verificar
cd project-directory/
npm run lint
npm run compile
npm test

Esta secuencia de comandos verifica la sintaxis del código de TypeScript, lo compila sin producir ningún resultado en el subdirectorio dist/ y ejecuta pruebas automatizadas desde test.ts.

Entrega
cd project-directory/
npm run start

Durante el desarrollo, esta secuencia de comandos entrega los paquetes de aplicaciones para los entornos de ejecución de Chrome y Node.js de forma local.

Implementa el controlador IDENTIFY

El controlador IDENTIFY se activará cuando el dispositivo Google Home o Google Nest se reinicie y detecte dispositivos locales sin verificar (incluidos los dispositivos finales conectados a un concentrador). La plataforma de Local Home buscará dispositivos locales con la información de configuración de análisis que especificaste anteriormente y llamará a tu controlador IDENTIFY con los resultados del análisis.

El IdentifyRequest de la plataforma de Local Home contiene los datos de análisis de una instancia de LocalIdentifiedDevice. Solo se propaga una instancia device, en función de la configuración de análisis que descubrió el dispositivo.

Si los resultados del análisis coinciden con tu dispositivo, el controlador IDENTIFY debe mostrar un objeto IdentifyResponsePayload, que incluya un objeto device con metadatos de casa inteligente (como los tipos, las características y el estado del informe).

Google establece una asociación de dispositivo si el verificationId de la respuesta IDENTIFY coincide con uno de los valores otherDeviceIds que muestra la respuesta SYNC.

Ejemplo

En los siguientes fragmentos, se muestra cómo puedes crear controladores IDENTIFY para integraciones de dispositivos independientes y concentradores, respectivamente.

Independiente
const identifyHandler = (request: IntentFlow.IdentifyRequest):
  IntentFlow.IdentifyResponse => {

    // Obtain scan data from protocol defined in your scan config
    const device = request.inputs[0].payload.device;
    if (device.udpScanData === undefined) {
      throw Error("Missing discovery response");
    }
    const scanData = device.udpScanData.data;

    // Decode scan data to obtain metadata about local device
    const verificationId = "local-device-id";

    // Return a response
    const response: IntentFlow.IdentifyResponse = {
      intent: Intents.IDENTIFY,
      requestId: request.requestId,
      payload: {
        device: {
          id: device.id || "",
          verificationId, // Must match otherDeviceIds in SYNC response
        },
      },
    };
    return response;
  };
Concentrador
const identifyHandler = (request: IntentFlow.IdentifyRequest):
  IntentFlow.IdentifyResponse => {

    // Obtain scan data from protocol defined in your scan config
    const device = request.inputs[0].payload.device;
    if (device.udpScanData === undefined) {
      throw Error("Missing discovery response");
    }
    const scanData = device.udpScanData.data;

    // Decode scan data to obtain metadata about local device
    const proxyDeviceId = "local-hub-id";

    // Return a response
    const response: IntentFlow.IdentifyResponse = {
      intent: Intents.IDENTIFY,
      requestId: request.requestId,
      payload: {
        device: {
          id: proxyDeviceId,
          isProxy: true,     // Device can control other local devices
          isLocalOnly: true, // Device not present in `SYNC` response
        },
      },
    };
    return response;
  };

Identifica dispositivos detrás de un concentrador

Si Google identifica un dispositivo concentrador, tratará al concentrador como el conducto hacia los dispositivos extremos conectados del concentrador y tratará de verificar esos dispositivos finales.

Para permitir que Google confirme que hay un dispositivo concentrador, sigue estas instrucciones para tu controlador IDENTIFY:

  • Si tu respuesta SYNC informa los IDs de los dispositivos finales locales conectados al concentrador, configura isProxy como true en IdentifyResponsePayload.
  • Si tu respuesta SYNC no informa el dispositivo concentrador, establece isLocalOnly como true en IdentifyResponsePayload.
  • El campo device.id contiene el ID de dispositivo local para el dispositivo concentrador.

Implementa el controlador REACHABLE_AVAILABLE (solo integraciones del concentrador)

Google envía el intent REACHABLE_DEVICES para confirmar qué dispositivos finales se pueden controlar de forma local. Este intent se activa cada vez que Google ejecuta un análisis de descubrimiento (aproximadamente una vez por minuto), siempre que se detecte que el concentrador está en línea.

Implementarás el controlador REACHABLE_DEVICES de manera similar al controlador IDENTIFY, excepto que el controlador debe recopilar IDs de dispositivos adicionales a los que puede acceder el dispositivo del proxy local (es decir, el concentrador). El campo device.verificationId contiene el ID de dispositivo local de un dispositivo final que está conectado al concentrador.

El ReachableDevicesRequest de la plataforma de Local Home contiene una instancia de LocalIdentifiedDevice. A través de esta instancia, puedes obtener el ID de dispositivo del proxy y los datos de los resultados del análisis.

El controlador REACHABLE_DEVICES debe mostrar un objeto ReachableDevicesPayload que incluya un objeto devices con un array de valores verificationId que representen los dispositivos finales que controla el concentrador. Los valores verificationId deben coincidir con uno de los otherDeviceIds de la respuesta SYNC.

En el siguiente fragmento, se muestra cómo crear tu controlador REACHABLE_DEVICES.

Concentrador
const reachableDevicesHandler = (request: IntentFlow.ReachableDevicesRequest):
  IntentFlow.ReachableDevicesResponse => {

    // Reference to the local proxy device
    const proxyDeviceId = request.inputs[0].payload.device.id;

    // Gather additional device ids reachable by local proxy device
    // ...

    const reachableDevices = [
      // Each verificationId must match one of the otherDeviceIds
      // in the SYNC response
      { verificationId: "local-device-id-1" },
      { verificationId: "local-device-id-2" },
    ];

    // Return a response
    const response: IntentFlow.ReachableDevicesResponse = {
      intent: Intents.REACHABLE_DEVICES,
      requestId: request.requestId,
      payload: {
        devices: reachableDevices,
      },
    };
    return response;
  };

Implementa el controlador EXECUTE

El controlador EXECUTE de la app procesa los comandos del usuario y usa el SDK de Local Home para acceder a tus dispositivos inteligentes a través de un protocolo existente.

La plataforma de Local Home pasa la misma carga útil de entrada a la función del controlador EXECUTE que para el intent EXECUTE a tu entrega en la nube. Del mismo modo, el controlador EXECUTE muestra datos de salida en el mismo formato en que se procesa el intent EXECUTE. Para simplificar la creación de respuestas, puedes usar la clase Execute.Response.Builder que proporciona el SDK de Local Home.

Tu app no tiene acceso directo a la dirección IP del dispositivo. En su lugar, usa la interfaz CommandRequest para crear comandos basados en uno de estos protocolos: UDP, TCP o HTTP. Luego, llama a la función deviceManager.send() para enviar los comandos.

Cuando orientes comandos a dispositivos, usa el ID de dispositivo (y los parámetros del campo customData, si están incluidos) de la respuesta SYNC para comunicarte con el dispositivo.

Ejemplo

En el siguiente fragmento de código, se muestra cómo crear el controlador EXECUTE.

Independiente/Concentrador
const executeHandler = (request: IntentFlow.ExecuteRequest):
  Promise<IntentFlow.ExecuteResponse> => {

    // Extract command(s) and device target(s) from request
    const command = request.inputs[0].payload.commands[0];
    const execution = command.execution[0];

    const response = new Execute.Response.Builder()
      .setRequestId(request.requestId);

    const result = command.devices.map((device) => {
      // Target id of the device provided in the SYNC response
      const deviceId = device.id;
      // Metadata for the device provided in the SYNC response
      // Use customData to provide additional required execution parameters
      const customData: any = device.customData;

      // Convert execution command into payload for local device
      let devicePayload: string;
      // ...

      // Construct a local device command over TCP
      const deviceCommand = new DataFlow.TcpRequestData();
      deviceCommand.requestId = request.requestId;
      deviceCommand.deviceId = deviceId;
      deviceCommand.data = devicePayload;
      deviceCommand.port = customData.port;
      deviceCommand.operation = Constants.TcpOperation.WRITE;

      // Send command to the local device
      return localHomeApp.getDeviceManager()
        .send(deviceCommand)
        .then((result) => {
          response.setSuccessState(result.deviceId, state);
        })
        .catch((err: IntentFlow.HandlerError) => {
          err.errorCode = err.errorCode || IntentFlow.ErrorCode.INVALID_REQUEST;
          response.setErrorState(device.id, err.errorCode);
        });
    });

    // Respond once all commands complete
    return Promise.all(result)
      .then(() => response.build());
  };

Implementa el controlador QUERY

El controlador QUERY de la app procesa las solicitudes de los usuarios y usa el SDK de Local Home para informar el estado de tus dispositivos inteligentes.

La plataforma de Local Home pasa la misma carga útil de solicitud a la función del controlador "QUERY" que el intent QUERY a tu entrega en la nube. De manera similar, el controlador QUERY muestra datos en el mismo formato en que se procesa el intent QUERY.

Enviar comandos a dispositivos que se encuentran detrás de un concentrador

Para controlar los dispositivos finales detrás de un concentrador, es posible que debas proporcionar información adicional en la carga útil del comando específico del protocolo que se envía al concentrador para que este identifique a qué dispositivo está dirigido el comando. En algunos casos, esto se puede inferir directamente del valor device.id, pero cuando este no sea el caso, debes incluir estos datos adicionales como parte del campo customData.

Si creaste tu app con TypeScript, recuerda compilarla en JavaScript. Puedes usar el sistema de módulos que prefieras para escribir el código. Asegúrate de que el destino sea compatible con el navegador Chrome.