diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000000000000000000000000000000000000..70fa045e816539292b653dad3df26c6f68b99be0
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1,2 @@
+!.eslintrc.js
+src/vendor/*
diff --git a/.eslintrc.js b/.eslintrc.js
new file mode 100644
index 0000000000000000000000000000000000000000..1cadfd103fd1910f79fb0e7928b0663b9e6a248a
--- /dev/null
+++ b/.eslintrc.js
@@ -0,0 +1,23 @@
+module.exports = {
+  parser: "babel-eslint",
+  env: {
+    browser: true,
+    es6: true,
+    node: true
+  },
+  globals: {
+    THREE: true,
+    AFRAME: true,
+    NAF: true
+  },
+  plugins: ["prettier", "react"],
+  rules: {
+    "prettier/prettier": "error",
+    "prefer-const": "error",
+    "no-var": "error",
+    "no-throw-literal": "error",
+    // Light console usage is useful but remove debug logs before merging to master.
+    "no-console": "off"
+  },
+  extends: ["prettier", "plugin:react/recommended", "eslint:recommended"]
+};
diff --git a/.eslintrc.json b/.eslintrc.json
deleted file mode 100644
index 8ab63d8cc0f6d8854b53e96c03a7889080ecb650..0000000000000000000000000000000000000000
--- a/.eslintrc.json
+++ /dev/null
@@ -1,16 +0,0 @@
-{
-  "parser": "babel-eslint",
-  "plugins": [
-    "prettier",
-    "react"
-  ],
-  "rules": {
-    "prettier/prettier": "error",
-    "prefer-const": "error",
-    "no-var": "error"
-  },
-  "extends": [
-    "prettier",
-    "plugin:react/recommended"
-  ]
-}
diff --git a/.prettierignore b/.prettierignore
index edded1b3e9cda6cef1bf33b9c8e08aa12f1f84a7..4bd6eed9e8bcf23ca49b895c57f0ba6d6290e14b 100644
--- a/.prettierignore
+++ b/.prettierignore
@@ -1,2 +1,3 @@
 package.json
 *.gltf
+src/vendor/
diff --git a/.travis.yml b/.travis.yml
new file mode 100644
index 0000000000000000000000000000000000000000..108e258b6953c22a31e4737b696cea02e5453112
--- /dev/null
+++ b/.travis.yml
@@ -0,0 +1,5 @@
+language: node_js
+node_js: node
+cache: yarn
+install: yarn 
+script: yarn lint
diff --git a/package.json b/package.json
index 5baab2abb42b937e578c0b379d96fcc81f9f9e39..71e501b591dd2af958da2361f34a2b32cb5fbc8b 100644
--- a/package.json
+++ b/package.json
@@ -11,7 +11,8 @@
     "postinstall": "node ./scripts/postinstall.js",
     "start": "cross-env NODE_ENV=development webpack-dev-server",
     "build": "rimraf ./public && cross-env NODE_ENV=production webpack --mode=production",
-    "prettier": "prettier --write src/**/*.js"
+    "prettier": "prettier --write '*.js' 'src/**/*.js'",
+    "lint": "eslint '*.js' 'src/**/*.js'"
   },
   "dependencies": {
     "@fortawesome/fontawesome": "^1.1.5",
diff --git a/src/activators/pressedmove.js b/src/activators/pressedmove.js
index c035383186e0997908be2eae5868a1a97e546dd4..fb7d7470723246b687049eafeb83d51e84f0e8bd 100644
--- a/src/activators/pressedmove.js
+++ b/src/activators/pressedmove.js
@@ -18,10 +18,10 @@ PressedMove.prototype = {
       this.onActivate(event);
     }
   },
-  onButtonDown: function(event) {
+  onButtonDown: function() {
     this.pressed = true;
   },
-  onButtonUp: function(event) {
+  onButtonUp: function() {
     this.pressed = false;
   },
 
diff --git a/src/activators/shortpress.js b/src/activators/shortpress.js
index 7f47450bef78161554a89569d4a610d7f7691877..fd9b8da0a212dc6acfba55509072d9c46c079305 100644
--- a/src/activators/shortpress.js
+++ b/src/activators/shortpress.js
@@ -16,13 +16,12 @@ function ShortPress(el, button, onActivate) {
 
 ShortPress.prototype = {
   onButtonDown(event) {
-    var self = this;
-    this.pressTimer = window.setTimeout(function() {
-      self.onActivate(event);
+    this.pressTimer = window.setTimeout(() => {
+      this.onActivate(event);
     }, this.timeOut);
   },
 
-  onButtonUp(event) {
+  onButtonUp() {
     clearTimeout(this.pressTimer);
   },
 
diff --git a/src/assets/avatars/avatars.js b/src/assets/avatars/avatars.js
index f7a5500048710c5bf0910e5bc781000764871312..7934ecb66c1439b04e892367dc3655d1f3826abe 100644
--- a/src/assets/avatars/avatars.js
+++ b/src/assets/avatars/avatars.js
@@ -1,65 +1,65 @@
 export const avatars = [
   {
-    "id": "botdefault",
-    "models": {
-      "low": `${ require("./BotDefault_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotDefault_Avatar.glb") }`
+    id: "botdefault",
+    models: {
+      low: `${require("./BotDefault_Avatar_Unlit.glb")}`,
+      high: `${require("./BotDefault_Avatar.glb")}`
     }
   },
   {
-    "id": "botbobo",
-    "models": {
-      "low": `${ require("./BotBobo_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotBobo_Avatar.glb") }`
+    id: "botbobo",
+    models: {
+      low: `${require("./BotBobo_Avatar_Unlit.glb")}`,
+      high: `${require("./BotBobo_Avatar.glb")}`
     }
   },
   {
-    "id": "botdom",
-    "models": {
-      "low": `${ require("./BotDom_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotDom_Avatar.glb") }`
+    id: "botdom",
+    models: {
+      low: `${require("./BotDom_Avatar_Unlit.glb")}`,
+      high: `${require("./BotDom_Avatar.glb")}`
     }
   },
   {
-    "id": "botgreg",
-    "models": {
-      "low": `${ require("./BotGreg_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotGreg_Avatar.glb") }`
+    id: "botgreg",
+    models: {
+      low: `${require("./BotGreg_Avatar_Unlit.glb")}`,
+      high: `${require("./BotGreg_Avatar.glb")}`
     }
   },
   {
-    "id": "botguest",
-    "models": {
-      "low": `${ require("./BotGuest_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotGuest_Avatar.glb") }`
+    id: "botguest",
+    models: {
+      low: `${require("./BotGuest_Avatar_Unlit.glb")}`,
+      high: `${require("./BotGuest_Avatar.glb")}`
     }
   },
   {
-    "id": "botjim",
-    "models": {
-      "low": `${ require("./BotJim_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotJim_Avatar.glb") }`
+    id: "botjim",
+    models: {
+      low: `${require("./BotJim_Avatar_Unlit.glb")}`,
+      high: `${require("./BotJim_Avatar.glb")}`
     }
   },
   {
-    "id": "botpinky",
-    "models": {
-      "low": `${ require("./BotPinky_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotPinky_Avatar.glb") }`
+    id: "botpinky",
+    models: {
+      low: `${require("./BotPinky_Avatar_Unlit.glb")}`,
+      high: `${require("./BotPinky_Avatar.glb")}`
     }
   },
   {
-    "id": "botrobert",
-    "models": {
-      "low": `${ require("./BotRobert_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotRobert_Avatar.glb") }`
+    id: "botrobert",
+    models: {
+      low: `${require("./BotRobert_Avatar_Unlit.glb")}`,
+      high: `${require("./BotRobert_Avatar.glb")}`
     }
   },
   {
-    "id": "botwoody",
-    "models": {
-      "low": `${ require("./BotWoody_Avatar_Unlit.glb") }`,
-      "high": `${ require("./BotWoody_Avatar.glb") }`
+    id: "botwoody",
+    models: {
+      low: `${require("./BotWoody_Avatar_Unlit.glb")}`,
+      high: `${require("./BotWoody_Avatar.glb")}`
     }
   }
 ];
diff --git a/src/avatar-selector.js b/src/avatar-selector.js
index 6acfe154caebb1c9f799631940aeb918e945fd7e..ea3ff71a0f1f225cb9ae9f68a813167125bc3207 100644
--- a/src/avatar-selector.js
+++ b/src/avatar-selector.js
@@ -1,7 +1,7 @@
 import ReactDOM from "react-dom";
 import React from "react";
 import queryString from "query-string";
-import { IntlProvider, FormattedMessage, addLocaleData } from "react-intl";
+import { IntlProvider, addLocaleData } from "react-intl";
 import en from "react-intl/locale-data/en";
 
 import "./assets/stylesheets/avatar-selector.scss";
@@ -12,13 +12,15 @@ import "./components/audio-feedback";
 import "./components/loop-animation";
 import "./elements/a-progressive-asset";
 import "./gltf-component-mappings";
-import { avatars } from "./assets/avatars/avatars.js";
-import { avatarIds } from "./utils/identity";
+import { avatars } from "./assets/avatars/avatars";
 
+import registerTelemetry from "./telemetry";
 import { App } from "./App";
 import AvatarSelector from "./react-components/avatar-selector";
 import localeData from "./assets/translations.data.json";
 
+registerTelemetry();
+
 window.APP = new App();
 const hash = queryString.parse(location.hash);
 const isMobile = AFRAME.utils.device.isMobile();
diff --git a/src/behaviours/oculus-touch-joystick-dpad4.js b/src/behaviours/oculus-touch-joystick-dpad4.js
index 758fb603c70faceb932539d722e08976e603046d..bf397ba2fbda2357c7fdf1cfc0310c8aa1ba548c 100644
--- a/src/behaviours/oculus-touch-joystick-dpad4.js
+++ b/src/behaviours/oculus-touch-joystick-dpad4.js
@@ -1,4 +1,4 @@
-import { angleTo4Direction, angleTo8Direction } from "../utils/dpad";
+import { angleTo4Direction } from "../utils/dpad";
 
 // @TODO specify 4 or 8 direction
 function oculus_touch_joystick_dpad4(el, outputPrefix) {
@@ -15,11 +15,8 @@ oculus_touch_joystick_dpad4.prototype = {
   emitDPad4: function(event) {
     const x = event.detail.axis[0];
     const y = event.detail.axis[1];
-    const inCenter =
-      Math.abs(x) < this.centerRadius && Math.abs(y) < this.centerRadius;
-    const current = inCenter
-      ? "center"
-      : this.angleToDirection(Math.atan2(x, -y));
+    const inCenter = Math.abs(x) < this.centerRadius && Math.abs(y) < this.centerRadius;
+    const current = inCenter ? "center" : this.angleToDirection(Math.atan2(x, -y));
     if (current !== this.previous) {
       this.previous = current;
       event.target.emit(`${this.outputPrefix}_dpad4_${current}`);
diff --git a/src/behaviours/vive-trackpad-dpad4.js b/src/behaviours/vive-trackpad-dpad4.js
index 3f4e3a275a4c6bad6a74aa6ef3c8836bb3a26500..99bdf8873d4cbcce3541691e6073af71c153db2b 100644
--- a/src/behaviours/vive-trackpad-dpad4.js
+++ b/src/behaviours/vive-trackpad-dpad4.js
@@ -1,4 +1,4 @@
-import { angleTo4Direction, angleTo8Direction } from "../utils/dpad";
+import { angleTo4Direction } from "../utils/dpad";
 
 function vive_trackpad_dpad4(el, outputPrefix) {
   this.outputPrefix = outputPrefix;
@@ -16,20 +16,17 @@ function vive_trackpad_dpad4(el, outputPrefix) {
 }
 
 vive_trackpad_dpad4.prototype = {
-  press: function(_) {
+  press: function() {
     this.pressed = true;
   },
-  unpress: function(_) {
+  unpress: function() {
     this.pressed = false;
   },
   emitDPad4: function(event) {
     const x = event.detail.axis[0];
     const y = event.detail.axis[1];
-    const inCenter =
-      Math.abs(x) < this.centerRadius && Math.abs(y) < this.centerRadius;
-    const direction = inCenter
-      ? "center"
-      : angleTo4Direction(Math.atan2(x, -y));
+    const inCenter = Math.abs(x) < this.centerRadius && Math.abs(y) < this.centerRadius;
+    const direction = inCenter ? "center" : angleTo4Direction(Math.atan2(x, -y));
     const pressed = this.pressed ? "pressed_" : "";
     const current = `${pressed + direction}`; // e.g. "pressed_north"
 
diff --git a/src/components/bone-mute-state-indicator.js b/src/components/bone-mute-state-indicator.js
index 7c6dc5913e971a3a3981f2d8af7ed3e151dcf0e8..2d79b3f9c37feee028b89afbe6eaffe7bbd654d8 100644
--- a/src/components/bone-mute-state-indicator.js
+++ b/src/components/bone-mute-state-indicator.js
@@ -12,12 +12,8 @@ AFRAME.registerComponent("bone-mute-state-indicator", {
   init() {
     this.onStateToggled = this.onStateToggled.bind(this);
     this.el.addEventListener("model-loaded", () => {
-      this.unmutedBone = this.el.object3D.getObjectByName(
-        this.data.unmutedBoneName
-      );
-      this.mutedBone = this.el.object3D.getObjectByName(
-        this.data.mutedBoneName
-      );
+      this.unmutedBone = this.el.object3D.getObjectByName(this.data.unmutedBoneName);
+      this.mutedBone = this.el.object3D.getObjectByName(this.data.mutedBoneName);
       console.log(this.unmutedBone, this.mutedBone);
       this.modelLoaded = true;
 
diff --git a/src/components/character-controller.js b/src/components/character-controller.js
index 20553d36ce9c06ca84806fd5b6adc7350ddf2aaa..1249dff3b0fbb3231bb52bd373e6cd91b7f3a544 100644
--- a/src/components/character-controller.js
+++ b/src/components/character-controller.js
@@ -60,11 +60,11 @@ AFRAME.registerComponent("character-controller", {
     this.angularVelocity = event.detail.value;
   },
 
-  snapRotateLeft: function(event) {
+  snapRotateLeft: function() {
     this.pendingSnapRotationMatrix.copy(this.leftRotationMatrix);
   },
 
-  snapRotateRight: function(event) {
+  snapRotateRight: function() {
     this.pendingSnapRotationMatrix.copy(this.rightRotationMatrix);
   },
 
@@ -79,9 +79,6 @@ AFRAME.registerComponent("character-controller", {
     const rotationInvMatrix = new THREE.Matrix4();
     const pivotRotationMatrix = new THREE.Matrix4();
     const pivotRotationInvMatrix = new THREE.Matrix4();
-    const position = new THREE.Vector3();
-    const currentPosition = new THREE.Vector3();
-    const movementVector = new THREE.Vector3();
 
     return function(t, dt) {
       const deltaSeconds = dt / 1000;
diff --git a/src/components/event-repeater.js b/src/components/event-repeater.js
index f099038b48ca692f1e7443e650b59cdfaeb7481e..64e28720698d8aaa26a8f0ce1525301cebd1ee2f 100644
--- a/src/components/event-repeater.js
+++ b/src/components/event-repeater.js
@@ -26,5 +26,4 @@ AFRAME.registerComponent("event-repeater", {
   _handleEvent: function(event, e) {
     this.el.emit(event, e.details);
   }
-
 });
diff --git a/src/components/haptic-feedback.js b/src/components/haptic-feedback.js
index 1ef413d38939a183c25438694cebaec934d9cf23..35a79cd9555e2864e5c496f671fcb9ddd8831345 100644
--- a/src/components/haptic-feedback.js
+++ b/src/components/haptic-feedback.js
@@ -18,9 +18,9 @@ AFRAME.registerComponent("haptic-feedback", {
   },
 
   getActuator() {
-    return new Promise((resolve, reject) => {
+    return new Promise(resolve => {
       const tryGetActivator = () => {
-        var trackedControls = this.el.components["tracked-controls"];
+        const trackedControls = this.el.components["tracked-controls"];
         if (
           trackedControls &&
           trackedControls.controller &&
@@ -44,7 +44,7 @@ AFRAME.registerComponent("haptic-feedback", {
   },
 
   pulse: function(event) {
-    let { intensity } = event.detail;
+    const { intensity } = event.detail;
     if (!strengthForIntensity[intensity]) {
       console.warn(`Invalid intensity : ${intensity}`);
       return;
diff --git a/src/components/ik-controller.js b/src/components/ik-controller.js
index 0a1990bd8ede46f4e5bbdde5cfced182ed4f5a3d..4fd89465e1d4ce97c24276b5d23d89eda5a25163 100644
--- a/src/components/ik-controller.js
+++ b/src/components/ik-controller.js
@@ -142,7 +142,6 @@ AFRAME.registerComponent("ik-controller", {
       cameraYRotation,
       cameraYQuaternion,
       invHipsQuaternion,
-      headQuaternion,
       leftHand,
       rightHand,
       rootToChest,
diff --git a/src/components/in-world-hud.js b/src/components/in-world-hud.js
index e69cfd7374eaa1ca1972b3cb00784d837dc987b8..2c10aad240d6ee2d21ea8d536751b0fc3785a463 100644
--- a/src/components/in-world-hud.js
+++ b/src/components/in-world-hud.js
@@ -96,7 +96,7 @@ AFRAME.registerComponent("in-world-hud", {
     this.el.sceneEl.removeEventListener("micAudio", this.onAudioFrequencyChange);
   },
 
-  tick: function(t, dt) {
+  tick: function() {
     if (!this.analyser) return;
 
     this.analyser.getByteFrequencyData(this.levels);
diff --git a/src/components/mute-mic.js b/src/components/mute-mic.js
index 272b40e3b41f5d4aa49d7107b6aa41785b8d5596..00729ba1904002a88729e21ed82f69dbb872dda9 100644
--- a/src/components/mute-mic.js
+++ b/src/components/mute-mic.js
@@ -1,6 +1,6 @@
 const bindAllEvents = function(elements, events, f) {
   if (!elements || !elements.length) return;
-  for (var el of elements) {
+  for (const el of elements) {
     events.length &&
       events.forEach(e => {
         el.addEventListener(e, f);
@@ -9,7 +9,7 @@ const bindAllEvents = function(elements, events, f) {
 };
 const unbindAllEvents = function(elements, events, f) {
   if (!elements || !elements.length) return;
-  for (var el of elements) {
+  for (const el of elements) {
     events.length &&
       events.forEach(e => {
         el.removeEventListener(e, f);
diff --git a/src/components/networked-counter.js b/src/components/networked-counter.js
index c35dcbaa461fa5ca8627d7cf693455d4244ec9fe..9c4fb7105578b49b48d5abe61e4c92193a0aaede 100644
--- a/src/components/networked-counter.js
+++ b/src/components/networked-counter.js
@@ -20,7 +20,7 @@ AFRAME.registerComponent("networked-counter", {
         item.el.removeEventListener(this.data.release_event, item.onReleaseHandler);
       }
     }
-    
+
     for (const id in this.timeouts) {
       this._removeTimeout(id);
     }
@@ -75,11 +75,11 @@ AFRAME.registerComponent("networked-counter", {
     }
   },
 
-  _onGrabbed: function(id, e) {
+  _onGrabbed: function(id) {
     this._removeTimeout(id);
   },
 
-  _onReleased: function(id, e) {
+  _onReleased: function(id) {
     this._removeTimeout(id);
     this._addTimeout(id);
     this.queue[id].ts = Date.now();
@@ -91,7 +91,6 @@ AFRAME.registerComponent("networked-counter", {
         ts = Number.MAX_VALUE;
       for (const id in this.queue) {
         if (this.queue.hasOwnProperty(id)) {
-          const expiration = this.queue[id].ts + this.data.ttl * 1000;
           if (this.queue[id].ts < ts && !this._isCurrentlyGrabbed(id)) {
             oldest = this.queue[id];
             ts = this.queue[id].ts;
diff --git a/src/components/networked-video-player.js b/src/components/networked-video-player.js
index 51adcf2f7f2ddd6d0023c54442e03627ea952c70..189a43e393f359a5c2d217b5978132476f6407c6 100644
--- a/src/components/networked-video-player.js
+++ b/src/components/networked-video-player.js
@@ -25,7 +25,7 @@ AFRAME.registerComponent("networked-video-player", {
     if (ownerId !== NAF.clientId && rejectScreenShares) {
       // Toggle material visibility since object visibility is network-synced
       // TODO: There ought to be a better way to disable network syncs on a remote entity
-      this.el.setAttribute("material", {visible: false});
+      this.el.setAttribute("material", { visible: false });
       return;
     }
 
diff --git a/src/components/offset-relative-to.js b/src/components/offset-relative-to.js
index 923eae3e5673488e8eceb726b9e91f592285f546..f940b687284b34677c579e931557fc9671631f41 100644
--- a/src/components/offset-relative-to.js
+++ b/src/components/offset-relative-to.js
@@ -12,10 +12,7 @@ AFRAME.registerComponent("offset-relative-to", {
   },
   init() {
     this.updateOffset();
-    this.el.sceneEl.addEventListener(
-      this.data.on,
-      this.updateOffset.bind(this)
-    );
+    this.el.sceneEl.addEventListener(this.data.on, this.updateOffset.bind(this));
   },
   updateOffset() {
     const offsetVector = new THREE.Vector3().copy(this.data.offset);
diff --git a/src/components/player-info.js b/src/components/player-info.js
index fc94e6092ae18bb654a5e344a37649ab41be9d11..b9455352a104bd71aceb07ed4718a2999a4d08b6 100644
--- a/src/components/player-info.js
+++ b/src/components/player-info.js
@@ -12,7 +12,7 @@ AFRAME.registerComponent("player-info", {
   pause() {
     this.el.removeEventListener("model-loaded", this.applyProperties);
   },
-  update(oldProps) {
+  update() {
     this.applyProperties();
   },
   applyProperties() {
diff --git a/src/components/super-cursor.js b/src/components/super-cursor.js
index e498a1acc2ad9b99c9131134bd17560bee1af531..28589171658f9d06ed57b4e604d91a73032ea583 100644
--- a/src/components/super-cursor.js
+++ b/src/components/super-cursor.js
@@ -103,7 +103,7 @@ AFRAME.registerComponent("super-cursor", {
     }
   },
 
-  _handleMouseDown: function(e) {
+  _handleMouseDown: function() {
     if (this.isInteractable) {
       const lookControls = this.data.camera.components["look-controls"];
       lookControls.pause();
@@ -115,7 +115,7 @@ AFRAME.registerComponent("super-cursor", {
     this.mousePos.set(e.clientX / window.innerWidth * 2 - 1, -(e.clientY / window.innerHeight) * 2 + 1);
   },
 
-  _handleMouseUp: function(e) {
+  _handleMouseUp: function() {
     const lookControls = this.data.camera.components["look-controls"];
     lookControls.play();
     this.data.cursor.emit("action_release", {});
@@ -125,13 +125,13 @@ AFRAME.registerComponent("super-cursor", {
     if (this.isGrabbing) this.currentDistanceMod += e.deltaY / 10;
   },
 
-  _handleEnterVR: function(e) {
+  _handleEnterVR: function() {
     if (AFRAME.utils.device.checkHeadsetConnected() || AFRAME.utils.device.isMobile()) {
       this._disable();
     }
   },
 
-  _handleExitVR: function(e) {
+  _handleExitVR: function() {
     this._enable();
   },
 
diff --git a/src/components/super-networked-interactable.js b/src/components/super-networked-interactable.js
index 4aeafecaa460379852a0ff4e9eb0f0f0d25bf9f3..a5baed53dd2f3f5404a81a26150904353434e9e9 100644
--- a/src/components/super-networked-interactable.js
+++ b/src/components/super-networked-interactable.js
@@ -11,7 +11,7 @@ AFRAME.registerComponent("super-networked-interactable", {
     NAF.utils.getNetworkedEntity(this.el).then(networkedEl => {
       this.networkedEl = networkedEl;
       if (!NAF.utils.isMine(networkedEl)) {
-        this.el.setAttribute("body", {type: "dynamic", mass: 0});
+        this.el.setAttribute("body", { type: "dynamic", mass: 0 });
       } else {
         this.counter.register(networkedEl);
       }
@@ -33,7 +33,7 @@ AFRAME.registerComponent("super-networked-interactable", {
     this.hand = e.detail.hand;
     if (this.networkedEl && !NAF.utils.isMine(this.networkedEl)) {
       if (NAF.utils.takeOwnership(this.networkedEl)) {
-        this.el.setAttribute("body", {mass: this.data.mass});
+        this.el.setAttribute("body", { mass: this.data.mass });
         this.counter.register(this.networkedEl);
       } else {
         this.el.emit("grab-end", { hand: this.hand });
@@ -42,8 +42,8 @@ AFRAME.registerComponent("super-networked-interactable", {
     }
   },
 
-  _onOwnershipLost: function(e) {
-    this.el.setAttribute("body", {mass: 0});
+  _onOwnershipLost: function() {
+    this.el.setAttribute("body", { mass: 0 });
     this.el.emit("grab-end", { hand: this.hand });
     this.hand = null;
     this.counter.deregister(this.el);
diff --git a/src/components/super-spawner.js b/src/components/super-spawner.js
index 72d24eff785b9eebed42c60e2e4feb21e659468a..bb6762a200eeacf53c29179f95554c7dc8fdab57 100644
--- a/src/components/super-spawner.js
+++ b/src/components/super-spawner.js
@@ -19,7 +19,7 @@ AFRAME.registerComponent("super-spawner", {
   },
 
   remove: function() {
-    for (let entity of this.entities.keys()) {
+    for (const entity of this.entities.keys()) {
       const data = this.entities.get(entity);
       entity.removeEventListener("componentinitialized", data.componentinInitializedListener);
       entity.removeEventListener("bodyloaded", data.bodyLoadedListener);
@@ -39,12 +39,12 @@ AFRAME.registerComponent("super-spawner", {
 
     this.entities.set(entity, {
       hand: hand,
-      componentInitialized: false, 
-      bodyLoaded: false, 
-      componentinInitializedListener: componentinInitializedListener, 
+      componentInitialized: false,
+      bodyLoaded: false,
+      componentinInitializedListener: componentinInitializedListener,
       bodyLoadedListener: bodyLoadedListener
     });
-    
+
     entity.addEventListener("componentinitialized", componentinInitializedListener);
     entity.addEventListener("body-loaded", bodyLoadedListener);
 
@@ -60,7 +60,7 @@ AFRAME.registerComponent("super-spawner", {
     }
   },
 
-  _handleBodyLoaded: function(entity, e) {
+  _handleBodyLoaded: function(entity) {
     this.entities.get(entity).bodyLoaded = true;
     this._emitEvents.call(this, entity);
   },
diff --git a/src/components/virtual-gamepad-controls.js b/src/components/virtual-gamepad-controls.js
index d399e38b29765280a677bb23289cdc3e4c2aff6b..d70219bf1e6374fefc6d6daaeb9e8e10bfc27fb8 100644
--- a/src/components/virtual-gamepad-controls.js
+++ b/src/components/virtual-gamepad-controls.js
@@ -1,10 +1,6 @@
 import nipplejs from "nipplejs";
 import styles from "./virtual-gamepad-controls.css";
 
-const THREE = AFRAME.THREE;
-const DEGREES = Math.PI / 180;
-const HALF_PI = Math.PI / 2;
-
 AFRAME.registerComponent("virtual-gamepad-controls", {
   schema: {},
 
diff --git a/src/components/wasd-to-analog2d.js b/src/components/wasd-to-analog2d.js
index a424cb893ab10c29575667f7c68539dd65e90689..35b68026bd121a35bb7f7c139d15423262475b86 100644
--- a/src/components/wasd-to-analog2d.js
+++ b/src/components/wasd-to-analog2d.js
@@ -44,7 +44,7 @@ AFRAME.registerComponent("wasd-to-analog2d", {
     this.keys[key] = down;
   },
 
-  tick: function(t, dt) {
+  tick: function() {
     this.target = [0, 0];
 
     for (const key in this.keys) {
diff --git a/src/components/water.js b/src/components/water.js
index 9152e548435ea8b184e9d1eb0f8cd4c2371217cc..b7f176131ea97f6f47fbbb31b27c59e80350b78b 100644
--- a/src/components/water.js
+++ b/src/components/water.js
@@ -17,24 +17,14 @@ function MobileWater(geometry, options) {
 
   options = options || {};
 
-  const clipBias = options.clipBias !== undefined ? options.clipBias : 0.0;
   const time = options.time !== undefined ? options.time : 0.0;
-  const normalSampler =
-    options.waterNormals !== undefined ? options.waterNormals : null;
+  const normalSampler = options.waterNormals !== undefined ? options.waterNormals : null;
   const sunDirection =
-    options.sunDirection !== undefined
-      ? options.sunDirection
-      : new THREE.Vector3(0.70707, 0.70707, 0.0);
-  const sunColor = new THREE.Color(
-    options.sunColor !== undefined ? options.sunColor : 0xffffff
-  );
-  const waterColor = new THREE.Color(
-    options.waterColor !== undefined ? options.waterColor : 0x7f7f7f
-  );
-  const eye =
-    options.eye !== undefined ? options.eye : new THREE.Vector3(0, 0, 0);
-  const distortionScale =
-    options.distortionScale !== undefined ? options.distortionScale : 20.0;
+    options.sunDirection !== undefined ? options.sunDirection : new THREE.Vector3(0.70707, 0.70707, 0.0);
+  const sunColor = new THREE.Color(options.sunColor !== undefined ? options.sunColor : 0xffffff);
+  const waterColor = new THREE.Color(options.waterColor !== undefined ? options.waterColor : 0x7f7f7f);
+  const eye = options.eye !== undefined ? options.eye : new THREE.Vector3(0, 0, 0);
+  const distortionScale = options.distortionScale !== undefined ? options.distortionScale : 20.0;
   const side = options.side !== undefined ? options.side : THREE.FrontSide;
   const fog = options.fog !== undefined ? options.fog : false;
 
diff --git a/src/elements/a-progressive-asset.js b/src/elements/a-progressive-asset.js
index 8bf922aa17b10ce6eec09baf413e93dcf6d871f1..db51b9b74484043a1f83847cc64d4ccc4bc6554c 100644
--- a/src/elements/a-progressive-asset.js
+++ b/src/elements/a-progressive-asset.js
@@ -33,7 +33,7 @@ AFRAME.registerElement("a-progressive-asset", {
           src = lowSrc;
         }
 
-        this.fileLoader.setResponseType(this.getAttribute("response-type") || inferResponseType(src));
+        this.fileLoader.setResponseType(this.getAttribute("response-type"));
         this.fileLoader.load(
           src,
           function handleOnLoad(response) {
diff --git a/src/hub.js b/src/hub.js
index 5ed6373c2f08fa2316f3e18a2d08c23e5874828e..a99ba8b0df4e9d310333c63416e34d73b7343f7b 100644
--- a/src/hub.js
+++ b/src/hub.js
@@ -220,6 +220,8 @@ function mountUI(scene) {
   const forcedVREntryType = qs.vr_entry_type || null;
   const enableScreenSharing = qsTruthy("enable_screen_sharing");
 
+  // TODO: Refactor to avoid using return value
+  /* eslint-disable react/no-render-return-value */
   const uiRoot = ReactDOM.render(
     <UIRoot
       {...{
@@ -235,6 +237,7 @@ function mountUI(scene) {
     />,
     document.getElementById("ui-root")
   );
+  /* eslint-enable react/no-render-return-value */
 
   return uiRoot;
 }
diff --git a/src/index.js b/src/index.js
index 2518b42dd56390078e56b04220ed780ab6abe707..198ee8d47378bd740fc9a2891cc6945447442b2a 100644
--- a/src/index.js
+++ b/src/index.js
@@ -4,4 +4,5 @@ import ReactDOM from "react-dom";
 import HomeRoot from "./react-components/home-root";
 import registerTelemetry from "./telemetry";
 
+registerTelemetry();
 ReactDOM.render(<HomeRoot />, document.getElementById("home-root"));
diff --git a/src/react-components/2d-hud.js b/src/react-components/2d-hud.js
index 73c23c82f39548e1770e8ba1ab8e9b44ab0affae..8d9223a072c97c670210e67e48434cfc58543d49 100644
--- a/src/react-components/2d-hud.js
+++ b/src/react-components/2d-hud.js
@@ -1,4 +1,4 @@
-import React, { Component } from "react";
+import React from "react";
 import PropTypes from "prop-types";
 import cx from "classnames";
 
diff --git a/src/react-components/auto-exit-warning.js b/src/react-components/auto-exit-warning.js
index 8663dcca2277d42456511adb212238d3bcdb936f..d8691a85182c7f8bbd3ee9a7472e428df9aa2d3b 100644
--- a/src/react-components/auto-exit-warning.js
+++ b/src/react-components/auto-exit-warning.js
@@ -1,4 +1,4 @@
-import React, { Component } from "react";
+import React from "react";
 import { FormattedMessage } from "react-intl";
 import PropTypes from "prop-types";
 
diff --git a/src/react-components/entry-buttons.js b/src/react-components/entry-buttons.js
index e567c6e1111f362ab4175087794af1f38a6a1c24..8ed8b171d6e7ec08d82d59af2c5314d9353a7910 100644
--- a/src/react-components/entry-buttons.js
+++ b/src/react-components/entry-buttons.js
@@ -1,28 +1,28 @@
-import React, { Component } from "react";
+import React from "react";
 import { FormattedMessage } from "react-intl";
 import PropTypes from "prop-types";
-import MobileDetect from 'mobile-detect';
+import MobileDetect from "mobile-detect";
 
-import MobileScreenEntryImg from '../assets/images/mobile_screen_entry.svg';
-import DesktopScreenEntryImg from '../assets/images/desktop_screen_entry.svg';
-import GenericVREntryImg from '../assets/images/generic_vr_entry.svg';
-import GearVREntryImg from '../assets/images/gearvr_entry.svg';
-import DaydreamEntyImg from '../assets/images/daydream_entry.svg';
+import MobileScreenEntryImg from "../assets/images/mobile_screen_entry.svg";
+import DesktopScreenEntryImg from "../assets/images/desktop_screen_entry.svg";
+import GenericVREntryImg from "../assets/images/generic_vr_entry.svg";
+import GearVREntryImg from "../assets/images/gearvr_entry.svg";
+import DaydreamEntyImg from "../assets/images/daydream_entry.svg";
 
 const mobiledetect = new MobileDetect(navigator.userAgent);
 
-const EntryButton = (props) => (
-  <div className="entry-button" onClick={ props.onClick }>
-    <img src={props.iconSrc} className="entry-button__icon"/>
+const EntryButton = props => (
+  <div className="entry-button" onClick={props.onClick}>
+    <img src={props.iconSrc} className="entry-button__icon" />
     <div className="entry-button__label">
       <div className="entry-button__label__contents">
         <span>
-          <FormattedMessage id={ props.prefixMessageId }/>
+          <FormattedMessage id={props.prefixMessageId} />
         </span>
         <span className="entry-button--bolded">
-          <FormattedMessage id={ props.mediumMessageId }/>
+          <FormattedMessage id={props.mediumMessageId} />
         </span>
-        { props.subtitle && (<div className="entry-button__subtitle">{props.subtitle}</div>) }
+        {props.subtitle && <div className="entry-button__subtitle">{props.subtitle}</div>}
       </div>
     </div>
   </div>
@@ -34,20 +34,20 @@ EntryButton.propTypes = {
   prefixMessageId: PropTypes.string,
   mediumMessageId: PropTypes.string,
   subtitle: PropTypes.string
-}
+};
 
-export const TwoDEntryButton = (props) => {
+export const TwoDEntryButton = props => {
   const entryButtonProps = {
     ...props,
     iconSrc: mobiledetect.mobile() ? MobileScreenEntryImg : DesktopScreenEntryImg,
     prefixMessageId: "entry.screen-prefix",
-    mediumMessageId: mobiledetect.mobile() ? "entry.mobile-screen" : "entry.desktop-screen" 
+    mediumMessageId: mobiledetect.mobile() ? "entry.mobile-screen" : "entry.desktop-screen"
   };
 
-  return (<EntryButton  {...entryButtonProps}/>);
-}
+  return <EntryButton {...entryButtonProps} />;
+};
 
-export const GenericEntryButton = (props) => {
+export const GenericEntryButton = props => {
   const entryButtonProps = {
     ...props,
     iconSrc: GenericVREntryImg,
@@ -55,10 +55,10 @@ export const GenericEntryButton = (props) => {
     mediumMessageId: "entry.generic-medium"
   };
 
-  return (<EntryButton {...entryButtonProps}/>);
+  return <EntryButton {...entryButtonProps} />;
 };
 
-export const GearVREntryButton = (props) => {
+export const GearVREntryButton = props => {
   const entryButtonProps = {
     ...props,
     iconSrc: GearVREntryImg,
@@ -66,10 +66,10 @@ export const GearVREntryButton = (props) => {
     mediumMessageId: "entry.gearvr-medium"
   };
 
-  return (<EntryButton  {...entryButtonProps}/>);
+  return <EntryButton {...entryButtonProps} />;
 };
 
-export const DaydreamEntryButton = (props) => {
+export const DaydreamEntryButton = props => {
   const entryButtonProps = {
     ...props,
     iconSrc: DaydreamEntyImg,
@@ -77,6 +77,5 @@ export const DaydreamEntryButton = (props) => {
     mediumMessageId: "entry.daydream-medium"
   };
 
-  return (<EntryButton  {...entryButtonProps}/>);
+  return <EntryButton {...entryButtonProps} />;
 };
-
diff --git a/src/react-components/home-root.js b/src/react-components/home-root.js
index 332904b9db9fd29ecb8b57f6e929cf07127d78f3..984a3d44c2fbb29973c3f1ed9dee4395d015bfa2 100644
--- a/src/react-components/home-root.js
+++ b/src/react-components/home-root.js
@@ -1,8 +1,6 @@
 import React, { Component } from "react";
 import PropTypes from "prop-types";
-import classNames from "classnames";
-import queryString from "query-string";
-import { IntlProvider, injectIntl, FormattedMessage, addLocaleData } from "react-intl";
+import { IntlProvider, FormattedMessage, addLocaleData } from "react-intl";
 import en from "react-intl/locale-data/en";
 import homeVideo from "../assets/video/home.webm";
 
diff --git a/src/react-components/profile-entry-panel.js b/src/react-components/profile-entry-panel.js
index b6dcee0df47cd7c32ca61d2656d951b954776ae4..fb92fc06d1f849292e6b68a0c905c51806f8e164 100644
--- a/src/react-components/profile-entry-panel.js
+++ b/src/react-components/profile-entry-panel.js
@@ -1,21 +1,22 @@
-import React, { Component } from 'react';
-import PropTypes from 'prop-types';
-import { injectIntl, FormattedMessage } from 'react-intl';
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { injectIntl, FormattedMessage } from "react-intl";
 import { SCHEMA } from "../storage/store";
 
 class ProfileEntryPanel extends Component {
   static propTypes = {
     store: PropTypes.object,
     messages: PropTypes.object,
-    finished: PropTypes.func
-  }
+    finished: PropTypes.func,
+    intl: PropTypes.object
+  };
 
   constructor(props) {
     super(props);
     window.store = this.props.store;
     this.state = {
       display_name: this.props.store.state.profile.display_name,
-      avatar_id: this.props.store.state.profile.avatar_id,
+      avatar_id: this.props.store.state.profile.avatar_id
     };
     this.props.store.addEventListener("statechanged", this.storeUpdated);
   }
@@ -23,67 +24,77 @@ class ProfileEntryPanel extends Component {
   storeUpdated = () => {
     const { avatar_id, display_name } = this.props.store.state.profile;
     this.setState({ avatar_id, display_name });
-  }
+  };
 
-  saveStateAndFinish = (e) => {
+  saveStateAndFinish = e => {
     e.preventDefault();
-    this.props.store.update({profile: {
-      display_name: this.state.display_name,
-      avatar_id: this.state.avatar_id
-    }});
+    this.props.store.update({
+      profile: {
+        display_name: this.state.display_name,
+        avatar_id: this.state.avatar_id
+      }
+    });
     this.props.finished();
-  }
+  };
 
-  stopPropagation = (e) => {
+  stopPropagation = e => {
     e.stopPropagation();
-  }
+  };
 
-  setAvatarStateFromIframeMessage = (e) => {
-    if (e.source !== this.avatarSelector.contentWindow) { return; }
-    this.setState({avatar_id: e.data.avatarId});
-  }
+  setAvatarStateFromIframeMessage = e => {
+    if (e.source !== this.avatarSelector.contentWindow) {
+      return;
+    }
+    this.setState({ avatar_id: e.data.avatarId });
+  };
 
   componentDidMount() {
     // stop propagation so that avatar doesn't move when wasd'ing during text input.
-    this.nameInput.addEventListener('keydown', this.stopPropagation);
-    this.nameInput.addEventListener('keypress', this.stopPropagation);
-    this.nameInput.addEventListener('keyup', this.stopPropagation);
-    window.addEventListener('message', this.setAvatarStateFromIframeMessage);
+    this.nameInput.addEventListener("keydown", this.stopPropagation);
+    this.nameInput.addEventListener("keypress", this.stopPropagation);
+    this.nameInput.addEventListener("keyup", this.stopPropagation);
+    window.addEventListener("message", this.setAvatarStateFromIframeMessage);
   }
 
   componentWillUnmount() {
-    this.props.store.removeEventListener('statechanged', this.storeUpdated);
-    this.nameInput.removeEventListener('keydown', this.stopPropagation);
-    this.nameInput.removeEventListener('keypress', this.stopPropagation);
-    this.nameInput.removeEventListener('keyup', this.stopPropagation);
-    window.removeEventListener('message', this.setAvatarStateFromIframeMessage);
+    this.props.store.removeEventListener("statechanged", this.storeUpdated);
+    this.nameInput.removeEventListener("keydown", this.stopPropagation);
+    this.nameInput.removeEventListener("keypress", this.stopPropagation);
+    this.nameInput.removeEventListener("keyup", this.stopPropagation);
+    window.removeEventListener("message", this.setAvatarStateFromIframeMessage);
   }
 
-  render () {
+  render() {
     const { formatMessage } = this.props.intl;
 
     return (
       <div className="profile-entry">
         <form onSubmit={this.saveStateAndFinish}>
-        <div className="profile-entry__box profile-entry__box--darkened">
-          <div className="profile-entry__subtitle">
-            <FormattedMessage id="profile.header"/>
+          <div className="profile-entry__box profile-entry__box--darkened">
+            <div className="profile-entry__subtitle">
+              <FormattedMessage id="profile.header" />
+            </div>
+            <input
+              className="profile-entry__form-field-text"
+              value={this.state.display_name}
+              onChange={e => this.setState({ display_name: e.target.value })}
+              required
+              pattern={SCHEMA.definitions.profile.properties.display_name.pattern}
+              title={formatMessage({ id: "profile.display_name.validation_warning" })}
+              ref={inp => (this.nameInput = inp)}
+            />
+            <iframe
+              className="profile-entry__avatar-selector"
+              src={
+                /* HACK: Have to account for the smoke test server like this. Feels wrong though. */
+                `/${/smoke/i.test(location.hostname) ? "smoke-" : ""}avatar-selector.html#avatar_id=${
+                  this.state.avatar_id
+                }`
+              }
+              ref={ifr => (this.avatarSelector = ifr)}
+            />
+            <input className="profile-entry__form-submit" type="submit" value={formatMessage({ id: "profile.save" })} />
           </div>
-          <input
-            className="profile-entry__form-field-text"
-            value={this.state.display_name} onChange={(e) => this.setState({display_name: e.target.value})}
-            required pattern={SCHEMA.definitions.profile.properties.display_name.pattern}
-            title={formatMessage({ id: "profile.display_name.validation_warning" })}
-            ref={inp => this.nameInput = inp}/>
-          <iframe
-            className="profile-entry__avatar-selector"
-            src={
-              /* HACK: Have to account for the smoke test server like this. Feels wrong though. */
-              `/${/smoke/i.test(location.hostname) ? 'smoke-' : ''}avatar-selector.html#avatar_id=${this.state.avatar_id}`
-            }
-            ref={ifr => this.avatarSelector = ifr}></iframe>
-          <input className="profile-entry__form-submit" type="submit" value={formatMessage({ id: "profile.save" }) }/>
-        </div>
         </form>
       </div>
     );
diff --git a/src/react-components/profile-info-header.js b/src/react-components/profile-info-header.js
index 6b9e82bb124e931dbf59b0867eaa3e36aae431f2..43ec49291007089d54e1d1a52c7a586e766e5fb4 100644
--- a/src/react-components/profile-info-header.js
+++ b/src/react-components/profile-info-header.js
@@ -1,4 +1,4 @@
-import React, { Component } from "react";
+import React from "react";
 import PropTypes from "prop-types";
 
 export const ProfileInfoHeader = props => (
diff --git a/src/react-components/ui-root.js b/src/react-components/ui-root.js
index 5c8918cd5646b779efd79295a42e0b38370c973a..3c64f2ae0384bda369387ab8bacd6fd427eca7e0 100644
--- a/src/react-components/ui-root.js
+++ b/src/react-components/ui-root.js
@@ -3,7 +3,6 @@ import PropTypes from "prop-types";
 import classNames from "classnames";
 import { VR_DEVICE_AVAILABILITY } from "../utils/vr-caps-detect";
 import queryString from "query-string";
-import { SCHEMA } from "../storage/store";
 import MobileDetect from "mobile-detect";
 import { IntlProvider, FormattedMessage, addLocaleData } from "react-intl";
 import en from "react-intl/locale-data/en";
@@ -34,13 +33,6 @@ const ENTRY_STEPS = {
   finished: "finished"
 };
 
-const HMD_MIC_REGEXES = [/\Wvive\W/i, /\Wrift\W/i];
-
-async function grantedMicLabels() {
-  const mediaDevices = await navigator.mediaDevices.enumerateDevices();
-  return mediaDevices.filter(d => d.label && d.kind === "audioinput").map(d => d.label);
-}
-
 // This is a list of regexes that match the microphone labels of HMDs.
 //
 // If entering VR mode, and if any of these regexes match an audio device,
@@ -49,13 +41,19 @@ async function grantedMicLabels() {
 //
 // Note that this doesn't have to be exhaustive: if no devices match any regex
 // then we rely upon the user to select the proper mic.
-const VR_DEVICE_MIC_LABEL_REGEXES = [];
+const HMD_MIC_REGEXES = [/\Wvive\W/i, /\Wrift\W/i];
+
+async function grantedMicLabels() {
+  const mediaDevices = await navigator.mediaDevices.enumerateDevices();
+  return mediaDevices.filter(d => d.label && d.kind === "audioinput").map(d => d.label);
+}
 
 const AUTO_EXIT_TIMER_SECONDS = 10;
 
 class UIRoot extends Component {
   static propTypes = {
     enterScene: PropTypes.func,
+    exitScene: PropTypes.func,
     concurrentLoadDetector: PropTypes.object,
     disableAutoExitOnConcurrentLoad: PropTypes.bool,
     forcedVREntryType: PropTypes.string,
@@ -253,32 +251,32 @@ class UIRoot extends Component {
     this.exit();
 
     // Launch via Oculus Browser
-    const qs = queryString.parse(document.location.search);
+    const location = window.location;
+    const qs = queryString.parse(location.search);
     qs.vr_entry_type = "gearvr"; // Auto-choose 'gearvr' after landing in Oculus Browser
 
-    const ovrwebUrl = `ovrweb://${document.location.protocol || "http:"}//${document.location.host}${document.location
-      .pathname || ""}?${queryString.stringify(qs)}#{document.location.hash || ""}`;
+    const ovrwebUrl =
+      `ovrweb://${location.protocol || "http:"}//${location.host}` +
+      `${location.pathname || ""}?${queryString.stringify(qs)}#${location.hash || ""}`;
 
-    document.location = ovrwebUrl;
+    window.location = ovrwebUrl;
   };
 
   enterDaydream = async () => {
-    const loc = document.location;
-
     if (this.state.availableVREntryTypes.daydream == VR_DEVICE_AVAILABILITY.maybe) {
       this.exit();
 
       // We are not in mobile chrome, so launch into chrome via an Intent URL
-      const qs = queryString.parse(document.location.search);
+      const location = window.location;
+      const qs = queryString.parse(location.search);
       qs.vr_entry_type = "daydream"; // Auto-choose 'daydream' after landing in chrome
 
-      const intentUrl = `intent://${document.location.host}${document.location.pathname || ""}?${queryString.stringify(
-        qs
-      )}#Intent;scheme=${(document.location.protocol || "http:").replace(
-        ":",
-        ""
-      )};action=android.intent.action.VIEW;package=com.android.chrome;end;`;
-      document.location = intentUrl;
+      const intentUrl =
+        `intent://${location.host}${location.pathname || ""}?` +
+        `${queryString.stringify(qs)}#Intent;scheme=${(location.protocol || "http:").replace(":", "")};` +
+        `action=android.intent.action.VIEW;package=com.android.chrome;end;`;
+
+      window.location = intentUrl;
     } else {
       await this.performDirectEntryFlow(true);
     }
@@ -327,7 +325,7 @@ class UIRoot extends Component {
     this.setState({ audioTrack: mediaStream.getAudioTracks()[0] });
   };
 
-  setupNewMediaStream = async constraints => {
+  setupNewMediaStream = async () => {
     const mediaStream = new MediaStream();
 
     // we should definitely have an audioTrack at this point.
diff --git a/src/storage/store.js b/src/storage/store.js
index 037e27bf1d9603a210423baa5ee6543c2ee77fdb..a517bfc5749c53eb9a193298194f7ed7fa0ee56f 100644
--- a/src/storage/store.js
+++ b/src/storage/store.js
@@ -4,7 +4,7 @@ import { Validator } from "jsonschema";
 const LOCAL_STORE_KEY = "___mozilla_duck";
 const STORE_STATE_CACHE_KEY = Symbol();
 const validator = new Validator();
-import { EventTarget } from "event-target-shim"
+import { EventTarget } from "event-target-shim";
 
 // Durable (via local-storage) schema-enforced state that is meant to be consumed via forward data flow.
 // (Think flux but with way less incidental complexity, at least for now :))
@@ -17,7 +17,7 @@ export const SCHEMA = {
       additionalProperties: false,
       properties: {
         display_name: { type: "string", pattern: "^[A-Za-z0-9-]{3,32}$" },
-        avatar_id: { type: "string" },
+        avatar_id: { type: "string" }
       }
     }
   },
@@ -26,11 +26,11 @@ export const SCHEMA = {
 
   properties: {
     id: { type: "string", pattern: "^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$" },
-    profile: { "$ref": "#/definitions/profile" },
+    profile: { $ref: "#/definitions/profile" }
   },
 
   additionalProperties: false
-}
+};
 
 export default class Store extends EventTarget {
   constructor() {
@@ -51,15 +51,14 @@ export default class Store extends EventTarget {
 
   update(newState) {
     if (newState.id) {
-      throw "Store id is immutable.";
+      throw new Error("Store id is immutable.");
     }
 
     const finalState = { ...this.state, ...newState };
     const isValid = validator.validate(finalState, SCHEMA).valid;
 
     if (!isValid) {
-      throw `Write of ${JSON.stringify(finalState)} to store failed schema validation.`;
-      return;
+      throw new Error(`Write of ${JSON.stringify(finalState)} to store failed schema validation.`);
     }
 
     localStorage.setItem(LOCAL_STORE_KEY, JSON.stringify(finalState));
diff --git a/src/systems/app-mode.js b/src/systems/app-mode.js
index 421bf48abbab344b877f9805803a7dfe30472336..1ef1c2cd0d7b79c75d0738717558901370320e5b 100644
--- a/src/systems/app-mode.js
+++ b/src/systems/app-mode.js
@@ -188,7 +188,7 @@ AFRAME.registerComponent("vr-mode-toggle-visibility", {
     this.el.sceneEl.removeEventListener("exit-vr", this.updateComponentState);
   },
 
-  updateComponentState(i) {
+  updateComponentState() {
     const inVRMode = this.el.sceneEl.is("vr-mode");
     this.el.setAttribute("visible", inVRMode !== this.data.invert);
   }
@@ -218,7 +218,7 @@ AFRAME.registerComponent("vr-mode-toggle-playing", {
     this.el.sceneEl.removeEventListener("exit-vr", this.updateComponentState);
   },
 
-  updateComponentState(i) {
+  updateComponentState() {
     const componentName = this.id;
     const inVRMode = this.el.sceneEl.is("vr-mode");
     this.el.components[componentName][inVRMode !== this.data.invert ? "play" : "pause"]();
diff --git a/src/systems/personal-space-bubble.js b/src/systems/personal-space-bubble.js
index 563818408f3dc4ea594f5ba2bc2eb10749dca5d2..e696705e138159a67c79e9707fe97bada64a9a58 100644
--- a/src/systems/personal-space-bubble.js
+++ b/src/systems/personal-space-bubble.js
@@ -39,16 +39,16 @@ AFRAME.registerSystem("personal-space-bubble", {
 
   tick() {
     // Update matrix positions once for each space bubble and space invader
-    for (var i = 0; i < this.bubbles.length; i++) {
+    for (let i = 0; i < this.bubbles.length; i++) {
       this.bubbles[i].object3D.updateMatrixWorld(true);
     }
 
-    for (var i = 0; i < this.invaders.length; i++) {
+    for (let i = 0; i < this.invaders.length; i++) {
       this.invaders[i].object3D.updateMatrixWorld(true);
     }
 
     // Loop through all of the space bubbles (usually one)
-    for (var i = 0; i < this.bubbles.length; i++) {
+    for (let i = 0; i < this.bubbles.length; i++) {
       const bubble = this.bubbles[i];
 
       bubblePos.setFromMatrixPosition(bubble.object3D.matrixWorld);
diff --git a/src/utils/concurrent-load-detector.js b/src/utils/concurrent-load-detector.js
index f05f619a3f51fab56a33acaec2eeb3b75e48ccea..53fc2d1fa37c1f236966211ba2abaaf4631960de 100644
--- a/src/utils/concurrent-load-detector.js
+++ b/src/utils/concurrent-load-detector.js
@@ -3,7 +3,7 @@
 // events.
 
 const LOCAL_STORE_KEY = "___concurrent_load_detector";
-import { EventTarget } from "event-target-shim"
+import { EventTarget } from "event-target-shim";
 
 export default class ConcurrentLoadDetector extends EventTarget {
   constructor(instanceKey) {
@@ -20,17 +20,17 @@ export default class ConcurrentLoadDetector extends EventTarget {
 
     // Check for concurrent load every second
     this.interval = setInterval(this._step, 1000);
-  }
+  };
 
   stop = () => {
     if (this.interval) {
       clearInterval(this.interval);
     }
-  }
+  };
 
   localStorageKey = () => {
     return `${LOCAL_STORE_KEY}_${this.instanceKey}`;
-  }
+  };
 
   _step = () => {
     const currentState = JSON.parse(localStorage.getItem(this.localStorageKey()));
@@ -40,5 +40,5 @@ export default class ConcurrentLoadDetector extends EventTarget {
       this.dispatchEvent(new CustomEvent("concurrentload"));
       this.stop();
     }
-  }
+  };
 }
diff --git a/src/utils/dpad.js b/src/utils/dpad.js
index 6f135122e6408bfdbc0a5a87d373104cbdc2e9bd..c603e75ad35438deb4311be5bdc68b64022da776 100644
--- a/src/utils/dpad.js
+++ b/src/utils/dpad.js
@@ -13,7 +13,7 @@ export function angleTo4Direction(angle) {
 
 export function angleTo8Direction(angle) {
   angle = (angle * THREE.Math.RAD2DEG + 180 + 45) % 360;
-  var direction = "";
+  let direction = "";
   if ((angle >= 0 && angle < 120) || angle >= 330) {
     direction += "north";
   }
diff --git a/src/utils/identity.js b/src/utils/identity.js
index 118fe30812c1013cb64c47a335c24465a12f6636..def830cbc4df4d9d1a71b0c10d7b54c5a5610b4a 100644
--- a/src/utils/identity.js
+++ b/src/utils/identity.js
@@ -164,7 +164,7 @@ const names = [
 ];
 
 function selectRandom(arr) {
-   return arr[Math.floor(Math.random() * arr.length)]
+  return arr[Math.floor(Math.random() * arr.length)];
 }
 
 export const avatarIds = avatars.map(av => av.id);
@@ -172,7 +172,7 @@ export const avatarIds = avatars.map(av => av.id);
 export function generateDefaultProfile() {
   const name = selectRandom(names);
   return {
-    display_name: name.replace(/^./, name[0].toUpperCase()) ,
+    display_name: name.replace(/^./, name[0].toUpperCase()),
     avatar_id: selectRandom(avatarIds)
   };
 }
diff --git a/webpack.config.js b/webpack.config.js
index 4cf14964c2deca54e3f83e4380d16e6facd36532..38903346a226a90efd2b714f9ed1c7d65f50002b 100644
--- a/webpack.config.js
+++ b/webpack.config.js
@@ -15,8 +15,6 @@ function createHTTPSConfig() {
     return false;
   }
 
-  let https;
-
   // Generate certs for the local webpack-dev-server.
   if (fs.existsSync(path.join(__dirname, "certs"))) {
     const key = fs.readFileSync(path.join(__dirname, "certs", "key.pem"));