Skip to content

Latest commit

 

History

History
333 lines (265 loc) · 10.9 KB

README.md

File metadata and controls

333 lines (265 loc) · 10.9 KB

Warning

This library is still in active development and not fully tested on all devices.

Divisible Workspaces Library

The Divisible Workspaces (DWS) Macro Library is a suite of tools which provides commands and a state management features for handling common requirements for divisbile workspaces built using Cisco RoomOS Collaboration Devices.

%%{
  init: {
    'theme': 'base',
    'themeVariables': {
      'primaryColor': '#89cff0',
      'primaryTextColor': '#fff',
      'primaryBorderColor': '#7C0000',
      'secondaryBorderColor': '#F8B229',
      'lineColor': '#F8B229',
      'secondaryColor': '#222222',
      'tertiaryColor': '#e1e0df',
      'tertiaryBorderColor': '#F8B229'
    }
  }
}%%
flowchart TB
    subgraph one [Combined]
        direction TB
        subgraph CombinedRoomAB [Room A+B]
            a1[Primary<br/>Codec]-->a2[Secondary<br/>Codec]
            a2[Secondary<br/>Codec]-->a1[Primary<br/>Codec]
        end
    end

    subgraph three [Divided]
        direction TB
        subgraph Room A
            b1[Primary<br/>Codec]
        end
         subgraph Room B
            b2[Secondary<br/>Codec]
        end
    end
Loading

Overview

This library aims to provide tools specific to divisbile workspaces use cases for Cisco Collaboration RoomOS Devices. Currently the following features are available:

  • Touch Panel Lock / Unlock:

    Easily Lock and Unlock the Touch Panel with a single command

    import DWS from './DWS_Lib';
    
    // Lock the Touch Panel
    DWS.Command.TouchPanel.Lock()
    
    // Unlock the Touch Panel
    DWS.Command.TouchPanel.Unlock()
  • Toggle Persistent Do Not Disturb:

    Enable or disable a persistant Do Not Disturb state

    import DWS from './DWS_Lib';
    
    // Activate Persistent Do Not Disturb
    DWS.Command.DoNotDisturb.Activate()
    
    // Deactivate Persistent Do Not Disturb
    DWS.Command.DoNotDisturb.Deactivate()
  • Toggle Ethernet Mute Using Stream Name:

    Easily mute and unmute an Ethernet Audio Input using just the stream name. No need to be concerned with streams being associated with different Ethernet Audio Input Connector IDs.

    import DWS from './DWS_Lib';
    
    // Mute Ethernet Audio Input with stream name 'streamName'
    DWS.Command.Audio.EthernetStream.Mute('streamName')
    
    // Unmute Ethernet Audio Input with stream name 'streamName'
    DWS.Command.Audio.EthernetStream.Unmute('streamName')
  • Request External Codec Status Check:

    Request a status from other Codecs based on their role name:

    import DWS from './DWS_Lib';
    import DWS from './DWS_Config';
    
    example():
    
    async function example(){
        // Setup Divisible Workspace Library with config
        // This contains the Codecs, their Roles and Credentials
        await DWS.Setup.Communication(config);
        try{
            // Request a status check from Codec with matching roles 'secondary'
            const secondaryStatus = await DWS.Command.RequestStatus('Secondary')
            console.log('Secondary Status:', secondaryStatus);
        } catch (error) {
            // Log Error if could not connect or did not receive response from Codec
            console.error(error)
        }
    }
    sequenceDiagram
    Primary Codec->>+Secondary Codec: Request Status Check
    Secondary Codec->>+Secondary Codec: Check if in call or presenting
    Secondary Codec->>+Primary Codec: Response With Result
    
    Loading
  • State Management:

    Easily define multiple states for each codec and switch between them with a single command. Additionally, you can define which xAPI Config/Status/Event subscription you wish to monitor only while in that state by adding them to the DWS.Subcriptions array and when changing between states, the subscriptions are all reset. Making it easy to have the codec react to events in one state and not another.

      import DWS from './DWS_Lib';
      import DWS from './DWS_Config';
    
      const states = {
          Primary: {
              Divided() {
                   // While 'Divided' don't monitor the VolumeMute xStatus
              },
              Combined() {
                  // While 'Combined' notify the 'Secondary' codec of VolumeMute changes
                  DWS.Subcriptions.push(xapi.Status.Audio.VolumeMute.on(state => {
                    DWS.NotifyCodecs(['Secondary'], 'volumeChange-'+state);
                    }))
              }
          },
          Secondary: {
              Divided() {
    
              },
              Combined() {
                  
              }
          }
      }
      
      example():
       
      async function example(){
          // Setup Divisible Workspace Library with config
          // This contains the Codecs, their Roles and Credentials
          await DWS.Setup.Config(config);
          // Setup all codec states
          await DWS.Setup.States(states);
          // Set the codec to a combined state
          await DWS.Command.ApplyState('combined');
      }
  • Send and Listen for Heartbeats:

    Send periodic heartbeats and listen for heartbeats, to and from specific devices by role name and trigger fallback states in the event no heartbeats have been received for a specified time.

    import DWS from './DWS_Lib';
    import DWS from './DWS_Config';
    
    const states = {
        Primary: {
            Divided() {
                // Stop sending heartbeats to 'Secondary'
                DWS.Command.Heartbeat.Send.Stop('Secondary');
                // Stop listening for hearbeats from 'Secondary'
                DWS.Command.Heartbeat.Listen.Stop('Secondary');
            },
            Combined() {
                // Listen for heartbeats from 'Secondary' and fallback to 'Divided'
                // if no heartbeats has been received for over 10 minutes
                DWS.Command.Heartbeat.Listen.Start('Secondary', 'Divided', 10);
    
                // Start Sending Heartbeats to 'Secondary' every minute
                DWS.Command.Heartbeat.Send.Start('Secondary', 1);
            }
        },
        Secondary: {
            Divided() {
                // Stop sending heartbeats to 'Primary'
                DWS.Command.Heartbeat.Send.Stop('Primary');
                // Stop listening for hearbeats from 'Primary'
                DWS.Command.Heartbeat.Listen.Stop('Primary');
            },
            Combined() {
                // Listen for heartbeats from 'Primary' and fallback to 'Divided'
                // if no heartbeats has been received for over 10 minutes
                DWS.Command.Heartbeat.Listen.Start('Primary', 'Divided', 10);
    
                // Start Sending Heartbeats to 'Primary' ever minute
                DWS.Command.Heartbeat.Send.Start('Primary', 1);
            }
        }
    }
    
    example():
     
    async function example(){
        // Setup Divisible Workspace Library with config
        // This contains the Codecs, their Roles and Credentials
        await DWS.Setup.Communication(config);
        // Setup all codec states
        await DWS.Setup.States(states);
        // Set the codec to a combined state
        await DWS.Command.ApplyState('combined');
    }
    sequenceDiagram
    Note over Primary Codec,Secondary Codec: Devices In "Combined" State
    box Blue Room A
    participant Primary Codec
    end
    box Red Room B
    participant Secondary Codec
    end
    par Start Listening For Heartbeats
        Primary Codec->>+Primary Codec: Start Listening
    and
        Secondary Codec->>+Secondary Codec: Start Listening
    end
    
    
    par Start Sending Heartbeats
        loop Send Heartbeat Every minute
            Primary Codec->>+Secondary Codec: heartbeat
        end
    and
       loop Send Heartbeat Every minute
        Secondary Codec->>+Primary Codec: heartbeat
        end
    end
    
    Note over Primary Codec,Secondary Codec: No Heartbeats Received For 10 Minutes
    
    par Trigger Fallback States
        Primary Codec->>+Primary Codec: Fallback To "Divided" State
    and 
        Secondary Codec->>+Secondary Codec: Fallback To "Divided" State
    end
    
    Loading

Setup

Prerequisites & Dependencies:

  • RoomOS/CE 11.8 or above Webex Device
  • Web admin access to the device to upload the macro
  • Network connectivity between your Webex Devices so they can communicate and sync state changes

Installation Steps:

  1. Download the DWS_Lib.js and DWS_Config.js macro files and upload it to your Webex Room devices Macro editor via the web interface.
  2. Configure the DWS_Config macro by specifying the Codecs involved in the divisible workspaces.
  3. Don't enable the DWS_Lib and DWS_Config macros as these are expected to be imported into you own macro.

Getting Started:

Refer to the DWS_Example macro which imports both the DWS_Lib and DWS_Config and demonstrates how to use the libraries state management and static methods features:

  • State Management:

    import xapi from 'xapi'
    import DWS from './DWS_Lib'
    import config from './DWS_Config'
    
    const states = {
        Primary: {
            Divided() {
                // Apply Primary Codec Divided changes here
            },
            Combined() {
                // Apply Primary Codec Combined changes here
            }
        },
        Secondary: {
            Divided() {
                // Apply Secondary Codec Divided changes here
            },
            Combined() {
                // Apply Secondary Codec Combined changes here
            }
        }
    }
    
    init();
    async function init(){
      console.log('Setting up Divisible Workspaces Library with config and states');
      await DWS.Setup.Communication(config);
      await DWS.Setup.States(states);   
      console.log('Setup Complete');
      setTimeout(DWS.Command.ApplyState, 2000, 'Combined')
    }
  • Static Methods:

    import DWS from './DWS_Lib';
    
    // Lock the Touch Panel
    DWS.Command.LockPanel()
    
    // Unlock the Touch Panel
    DWS.Command.UnlockPanel()

Demo

*For more demos & PoCs like this, check out our Webex Labs site.

License

All contents are licensed under the MIT license. Please see license for details.

Disclaimer

Everything included is for demo and Proof of Concept purposes only. Use of the site is solely at your own risk. This site may contain links to third party content, which we do not warrant, endorse, or assume liability for. These demos are for Cisco Webex use cases, but are not Official Cisco Webex Branded demos.

Questions

Please contact the WXSD team at [email protected] for questions. Or, if you're a Cisco internal employee, reach out to us on the Webex App via our bot ([email protected]). In the "Engagement Type" field, choose the "API/SDK Proof of Concept Integration Development" option to make sure you reach our team.