Simple Vue.js dashboard using JSXAPI WebSockets connection

You can also read additional information in my blog https://blogs.cisco.com/developer/dashboard-webex-device-monitoring

Requirements

Installation

git clone https://github.com/CiscoDevNet/sample-ws-jsxapi-vue
cd sample-ws-jsxapi-vue

install dependencies

npm install

serve with hot reload at localhost:8080

npm run dev

build for production with minification

npm run build

or using yarn

# yarn install is used to install all dependencies for a project
yarn install

serve with hot reload at localhost:8080

yarn run dev

build for production with minification

yarn run build

Using step-by-step instructions below you will create a simple Vue app.
In this App you create and template and Vue js methods that can process data from Webex devices.
We also create a WebSocket connection and track different types of events.

When you see a box that looks like this:

CLI command /OR/ JS code /OR/ HTML

You can read description of this block and copy this text and paste it into the IDE/terminal.

Here is a brief cheat sheet:
For Linux:

Copy - CTRL+INSERT

Paste - SHIFT+INSERT

For Mac:

Copy - CMD+C

Paste - CMD+V

For Windows:

Copy - CTRL+C

Paste - CTRL+V

Note all code is in current directory. You can compare your code and final version. In case of errors also refer to the code located in the current directory.

First step

Install Node

Install Vue-cli. Using Vue-cli you can start the project with some of the official Vue project templates or one of the many of the open-source Vue templates and, of course, you can create your own one and use it anywhere

npm install -g vue-cli

A simple Vue 2.0 Webpack & vue-loader setup for quick prototyping. Note this template is not suitable for production - for that you may want to use the full webpack template.

vue init webpack-simple vue-spa

Enter information Project name, description, Author, License, Using sass or Press Enter for choosing default variants.

cd vue-spa/
npm install

Vue component

There is the index.html file with a simple HTML markup including only the element with identifier app in a body. It will be replaced by a vue-generated DOM. That is the reason why you should not use the tag body as as a root element.

In the src directory we have the main.js file which is the entry point for webpack. The Vue components are imported here. Also, here we have a root Vue instance that has two properties for now. The property el provides the Vue instance with an existing DOM element to mount on. Another one is a render function which generates DOM from App.vue. In general, this is all we need to know about the structure of the webpack-simple template, not so much, isn’t it? The main part of our Vue app will be coded in App.vue. The .vue extension indicates that this file is a single-file vue component. It is one of the Vue’s features, let's get to know it better.

Install jsxapi package.
It's set of tools to integrate with the Cisco Telepresence Endpoint APIs in JavaScript.

npm install --save-dev jsxapi

Install Vuetify.js (Vue Material Design Component Framework)

npm install vuetify

For importing font and icons open index.html and insert in <head> </head> tag

<link href=’https://fonts.googleapis.com/css?family=Roboto:300,400,500,700|Material+Icons' rel=”stylesheet”>
<link href="https://cdn.jsdelivr.net/npm/@mdi/font@4.x/css/materialdesignicons.min.css" rel="stylesheet">

Import Vuetify. In src directory open main.js and insert code below after this line import App from './App.vue'

import Vuetify from 'vuetify'
import 'vuetify/dist/vuetify.min.css'
Vue.use(Vuetify)

Edit configuration and enable using Websocket

Using Webex device GUI you need to make next changes
Go to Setup -> Configurations
In left menu chose HttpClient settings and change AllowInsecureHTTPS from False on True

Then navigate to your Webex device GUI page like https://10.10.20.30 and approve the room kit self-sign certificate in your browser

WebSocket's library that we use will indicate an error when we using HTTP

Also, Using Webex device GUI you need to enable using Websocket

Go to Setup -> Configurations

In left menu chose NetworkServices settings and change Websocket from off on FollowHTTPService

Edit App.vue

In src directory open App.vue file. Inside <template> </template> <main> </main> tag insert
to create form like this

We use standart form sample

CLICK ME

      <v-form v-model="valid" ref="form">
        <v-container>
          <v-row>
            <v-col
              cols="12"
              md="4"
            >
              <v-text-field
                v-model="login"
                :rules="nameRules"
                :counter="10"
                label="Login"
                required
              ></v-text-field>
            </v-col>

        &lt;v-col
          cols="12"
          md="4"
        &gt;
          &lt;v-text-field
            v-model="password"
            :rules="nameRules"
            :counter="10"
            label="Password"
            required
          &gt;&lt;/v-text-field&gt;
        &lt;/v-col&gt;

        &lt;v-col
          cols="12"
          md="4"
        &gt;
          &lt;v-text-field
            v-model="ip"
            :rules="ipRules"
            label="Webex device IP"
            required
          &gt;&lt;/v-text-field&gt;
        &lt;/v-col&gt;
        &lt;v-btn
          @click="submit"
          :disabled="!fieldIsEmpty || !valid"
          v-if="!progress"&gt;
          submit
        &lt;/v-btn&gt;
        &lt;v-btn
          @click="close"
          :disabled="!connection"
          v-if="!progress"&gt;
          Close connection
        &lt;/v-btn&gt;
      &lt;/v-row&gt;
    &lt;/v-container&gt;
  &lt;/v-form&gt;

After this, we create two cards for collecting and displaying information.
In one of the cards, we insert form for a demo of WebSockets connection.
We can change the System Unit Name for testing online tracking this type of change.

CLICK ME

      <v-container class="grey lighten-5">
        <v-row no-gutters>
          <v-col
          >
            <v-card
              class="pa-2"
              outlined
              tile
            >
              <v-card
                max-width="375"
                class="mx-auto"
              >
                <v-list two-line>

              &lt;v-list-item @click=""&gt;
                &lt;v-list-item-icon&gt;
                  &lt;v-icon color="indigo"&gt;mdi-card-bulleted-outline&lt;/v-icon&gt;
                &lt;/v-list-item-icon&gt;

                &lt;v-list-item-content&gt;
                  &lt;v-list-item-title&gt;{{this.SystemName}}&lt;/v-list-item-title&gt;
                  &lt;v-list-item-subtitle&gt;System Name&lt;/v-list-item-subtitle&gt;
                &lt;/v-list-item-content&gt;

              &lt;/v-list-item&gt;

              &lt;v-list-item @click=""&gt;
                &lt;v-list-item-icon&gt;
                  &lt;v-icon color="indigo"&gt;mdi-alarm&lt;/v-icon&gt;
                &lt;/v-list-item-icon&gt;

                &lt;v-list-item-content&gt;
                  &lt;v-list-item-title&gt;{{this.dateTime}}&lt;/v-list-item-title&gt;
                  &lt;v-list-item-subtitle&gt;Date and Time&lt;/v-list-item-subtitle&gt;
                &lt;/v-list-item-content&gt;

              &lt;/v-list-item&gt;

              &lt;v-list-item @click=""&gt;
                &lt;v-list-item-icon&gt;
                  &lt;v-icon color="indigo"&gt;mdi-blur&lt;/v-icon&gt;
                &lt;/v-list-item-icon&gt;

                &lt;v-list-item-content&gt;
                  &lt;v-list-item-title&gt;{{this.netMac}}&lt;/v-list-item-title&gt;
                  &lt;v-list-item-subtitle&gt;MAC Address&lt;/v-list-item-subtitle&gt;
                &lt;/v-list-item-content&gt;

              &lt;/v-list-item&gt;

              &lt;v-divider inset&gt;&lt;/v-divider&gt;

              &lt;v-list-item @click=""&gt;
                &lt;v-list-item-icon&gt;
                  &lt;v-icon color="indigo"&gt;mdi-album&lt;/v-icon&gt;
                &lt;/v-list-item-icon&gt;

                &lt;v-list-item-content&gt;
                  &lt;v-list-item-title&gt;{{this.ipv4}}&lt;/v-list-item-title&gt;
                  &lt;v-list-item-subtitle&gt;IPv4&lt;/v-list-item-subtitle&gt;
                &lt;/v-list-item-content&gt;
              &lt;/v-list-item&gt;

              &lt;v-list-item @click=""&gt;
                &lt;v-list-item-icon&gt;
                  &lt;v-icon color="indigo"&gt;mdi-cloud-download-outline&lt;/v-icon&gt;
                &lt;/v-list-item-icon&gt;

                &lt;v-list-item-content&gt;
                  &lt;v-list-item-title&gt;{{this.SoftUpStat}}&lt;/v-list-item-title&gt;
                  &lt;v-list-item-subtitle&gt;Software Upgrade Status&lt;/v-list-item-subtitle&gt;
                &lt;/v-list-item-content&gt;
              &lt;/v-list-item&gt;

              &lt;v-divider inset&gt;&lt;/v-divider&gt;

              &lt;v-list-item @click=""&gt;
                &lt;v-list-item-icon&gt;
                  &lt;v-icon color="indigo"&gt;mdi-rename-box&lt;/v-icon&gt;
                &lt;/v-list-item-icon&gt;

                &lt;v-list-item-content&gt;
                  &lt;v-list-item-title&gt;{{this.SoftDispName}}&lt;/v-list-item-title&gt;
                  &lt;v-list-item-subtitle&gt;SystemUnit Software DisplayName&lt;/v-list-item-subtitle&gt;
                &lt;/v-list-item-content&gt;
              &lt;/v-list-item&gt;
            &lt;/v-list&gt;
          &lt;/v-card&gt;
        &lt;/v-card&gt;
      &lt;/v-col&gt;
      &lt;v-col
      &gt;
        &lt;v-card
          class="pa-2"
          outlined
          tile
        &gt;
          &lt;v-card
            max-width="375"
            class="mx-auto"
          &gt;
            &lt;v-text-field
              v-model="newSystemName"
              label="System Unit Name"
            &gt;&lt;/v-text-field&gt;
            &lt;v-btn
              @click="setname"
              :disabled="!connection"
              v-if="!progress"&gt;
              Set System Unit Name
            &lt;/v-btn&gt;
            &lt;v-list-item @click=""&gt;
              &lt;v-list-item-icon&gt;
                &lt;v-icon color="indigo"&gt;mdi-phone-in-talk-outline&lt;/v-icon&gt;
              &lt;/v-list-item-icon&gt;

              &lt;v-list-item-content&gt;
                &lt;v-list-item-title&gt;{{this.NumberOfActiveCalls}}&lt;/v-list-item-title&gt;
                &lt;v-list-item-subtitle&gt;Number Of Active Calls&lt;/v-list-item-subtitle&gt;
              &lt;/v-list-item-content&gt;
            &lt;/v-list-item&gt;
          &lt;/v-card&gt;
        &lt;/v-card&gt;
      &lt;/v-col&gt;
    &lt;/v-row&gt;
  &lt;/v-container&gt;

Inside <script> </script> tag

Import jsxapi module

const jsxapi = require('jsxapi');

All code snippets below you need insert in export default {INSERT_CODE}
Then paste the block

    data() {
      return {
        // the block where we initialize the variables for further reassignment in functions
        wsconnection: null,
        feedbackGroup: null,
        fireAction: null,
        historyEntries: [],
        connection: false,
        dateTime: [],
        netMac: null,
        ipv4: null,
        SoftUpStat: null,
        SoftDispName: null,
        NumberOfActiveCalls: null,
        valid: false,
        SystemName: '',
        newSystemName: '',
        login: '',
        password: '',
        nameRules: [
          v => !!v || 'Login is required',
          v => v.length <= 10 || 'Name must be less than 10 characters',
        ],
        ip: '',
        ipRules: [
          v => !!v || 'IP is required'
        ],
        progress: false
      }
    },

After this block edits the computed property.
A computed property is where the variable 'fieldIsEmpty' can take on the value of 'true' or 'false',
depending on whether the fields are filled or unfilled, respectively.

    computed: {
      fieldIsEmpty: function () {
        return !!(this.login && this.password && this.ip)
      },
    },

Create methods and functions

    methods: {
      // when users clicked on the 'submit' button function below will be called
      submit() {
        this.initConnection();
      },
      // when users clicked on the 'close connection' button function below will be called, and we'll clean variables and call function for close WebSockets connection
      close() {
        this.login = '';
        this.password = '';
        this.ip = '';
        this.closeConnection();
      },
      // New style API
      setname() {
        if (this.wsconnection) {
          this.wsconnection.Config.SystemUnit.Name.set(this.newSystemName);
          // xConfiguration SystemUnit Name: "Kit 006"
        }
      },
      initConnection() {
        this.ip = 'wss://'.concat(this.ip); // string concatenation for creating WebSocket URI
        console.log(this.ip);
        this.wsconnection = jsxapi.connect(this.ip, {username: this.login, password: this.password}); // initializing WebSockets connection
        this.connection = true;


    // Run functions for get information
    this.wsconnection.on('ready', () =&gt; {
      this.getCallHistory();
      this.getDateTime();
      this.getNetworkMac();
      this.getNetworkIpv4();
      this.getSoftUpStat();
      this.getSoftDispName();
      this.getNumberOfActiveCalls();
      this.getSystemName();
    });
    // Bundle feedback listeners for easy unsubscription
    // https://cisco-ce.github.io/jsxapi/classes/feedback.html
    this.feedbackGroup = this.wsconnection.feedback.group([
      // Register listener for track changing UserInterface Name
      this.wsconnection.status.on('UserInterface/ContactInfo/Name', (newName) =&gt; {
        console.log('SystemUnit Name', newName);
        this.SystemName = newName;
      }),
      // More about Feedback mechanism pages 78-79
      // https://www.cisco.com/c/dam/en/us/td/docs/telepresence/endpoint/ce910/collaboration-endpoint-software-api-reference-guide-ce910-temp.pdf
      this.wsconnection.status.on('SystemUnit/State/NumberOfActiveCalls', (newNumberCalls) =&gt; {
        console.log('NumberOfActiveCalls', newNumberCalls);
        this.NumberOfActiveCalls = newNumberCalls;
      })
    ]);

  },
  closeConnection() {
    this.wsconnection.close();
    // Disable feedback listening for all listeners of the group.
    this.feedbackGroup.off();
    console.log('Connection closed');
  },
  // Block where we call the functions to get the data
  // xapi has xConfiguration, xCommand and xStatus commands, for use it you need different methods
  // all list of available commands you can find in API reference guide
  // https://www.cisco.com/c/dam/en/us/td/docs/telepresence/endpoint/ce910/collaboration-endpoint-software-api-reference-guide-ce910-temp.pdf


  // in finction below we used OLD style API. So you can copy command like 'Provisioning Software UpgradeStatus Status'
  // and find it in API reference guide
  getCallHistory() {
    if (this.wsconnection) {
      this.wsconnection.command("CallHistory Get", {
        DetailLevel: "Full"
      })
        .then((history) =&gt; {
          if (history.status === 'OK') {
            this.historyEntries = history.Entry
          }
        });
    }
  },
  getDateTime() {
    if (this.wsconnection) {
      this.wsconnection.command("Time DateTime Get")
        .then((history) =&gt; {
          if (history.status === 'OK') {
            this.dateTime = [history.Day, history.Hour, history.Minute, history.Month, history.Second, history.Year]
          }
        });
    }
  },
  getNetworkMac() {
    if (this.wsconnection) {
      this.wsconnection.status.get("Network 1 Ethernet MacAddress")
        .then((history) =&gt; {
          this.netMac = history
        });
    }
  },
  getNetworkIpv4() {
    if (this.wsconnection) {
      this.wsconnection.status.get("Network 1 IPv4 Address")
        .then((history) =&gt; {
          this.ipv4 = history
        });
    }
  },
  getSoftUpStat() {
    if (this.wsconnection) {
      this.wsconnection.status.get("Provisioning Software UpgradeStatus Status")
        .then((history) =&gt; {
          this.SoftUpStat = history
        });
    }
  },
  getSoftDispName() {
    if (this.wsconnection) {
      this.wsconnection.status.get("SystemUnit Software DisplayName")
        .then((history) =&gt; {
          this.SoftDispName = history
        });
    }
  },
  // You can use sip addresses below for testing (automatically enabled a call):
  // 111@bjn.vc, fireplace@ivr.vc, goldfish@selfie.vc, halloween@ivr.vc, havnen@expressway.dk
  getNumberOfActiveCalls() {
    if (this.wsconnection) {
      this.wsconnection.status.get("SystemUnit State NumberOfActiveCalls")
        .then((history) =&gt; {
          this.NumberOfActiveCalls = history
        });
    }
  },
  getSystemName() {
    if (this.wsconnection) {
      this.wsconnection.status.get("UserInterface ContactInfo Name")
        .then((history) =&gt; {
          this.SystemName = history
        });
    }
  },
}

Run the App

npm run dev

To aggregate and visualize information, you can use the following templates (some of them):

Other Useful links

To access some sandbox/services you will need to use Cisco AnyConnect:

  • For Mac and Windows, you can download here
  • For Linux, install OpenConnect, see here

Use Case

Dashboard for Webex Device Monitoring

This Use Case can help you to create a dashboard to monitor and manage Webex devices.

The rapid growth in demand for remote work systems has resulted in new challenges of monitoring and troubleshooting devices used for this purpose. Recently, Webex events (calls, messages, etc) have jumped from 39 billion to 270 billion per day.

All well known video services are under heavy load. Cisco’s advantage in unified communication solutions is also based on high-quality functional devices equipped with high-quality cameras, sound systems, and microphones as well. Many employees started working remotely. Someone uses laptops for video conferencing; many employees make use of advanced devices, such as Webex devices: Series DX, MX, SX; Room/Roomkit/Codec and Webex Board’s.

RoomKit Sandbox

Source code and instruction

Other Useful links

View code on GitHub

Code Exchange Community

Get help, share code, and collaborate with other developers in the Code Exchange community.View Community
Disclaimer:
Cisco provides Code Exchange for convenience and informational purposes only, with no support of any kind. This page contains information and links from third-party websites that are governed by their own separate terms. Reference to a project or contributor on this page does not imply any affiliation with or endorsement by Cisco.