declare var chrome: any;
declare class TextDecoder {
  constructor(encoding: string);
  decode(data: DataView): string;
}

// A list of: {
//   id: number;
//   path: string;
// } where [id] is the [connectionId] (internal to Chrome) and [path] is the
// OS' name for the device (e.g. "COM4").
interface Connection {
  id: string;
  path: string;
}
let connections: Connection[] = [];

// A list of "ports", i.e. connected clients (such as web pages). Multiple web
// pages can connect to our service: they all receive the same data.
let ports = [];

interface Message {
  type: string;
  data: string;
  id: string;
}

function byPath(path: string): Connection[] {
  return connections.filter((x) => x.path == path);
}

function byId(id: string): Connection[] {
  return connections.filter((x) => x.id == id);
}

function onReceive(data: ArrayBuffer, id: string) {
  if (ports.length == 0) return;

  let view = new DataView(data);
  let decoder = new TextDecoder("utf-8");
  let decodedString = decoder.decode(view);
  ports.forEach(port => port.postMessage(<Message>{
    type: "serial",
    data: decodedString,
    id: id
  }));
}

function findNewDevices() {
  chrome.serial.getDevices(function (serialPorts) {
    serialPorts.forEach(function (serialPort) {
      if (byPath(serialPort.path).length == 0 &&
        serialPort.displayName == "mbed Serial Port") {
        try {
          chrome.serial.connect(serialPort.path, { bitrate: 115200 }, function (info) {
            // In case the [connect] operation takes more than five seconds...
            if (info && byPath(serialPort.path).length == 0)
              connections.push({
                id: info.connectionId,
                path: serialPort.path
              });
          });
        } catch (e) {
          console.log(`failed to connect to ${serialPort.displayName}`)
        }
      }
    });
  });
}

function main() {
  // Register new clients in the [ports] global variable.
  chrome.runtime.onConnectExternal.addListener(function (port) {
    console.log('connection to port ' + port.name)
    if (/^serial$/.test(port.name)) {
      ports.push(port);
      port.onDisconnect.addListener(function () {
        ports = ports.filter(function (x) { return x != port });
      });
    }
  });

  // When receiving data for one of the connections that we're tracking, forward
  // it to all connected clients.
  chrome.serial.onReceive.addListener(function (info) {
    if (byId(info.connectionId).length > 0)
      onReceive(info.data, info.connectionId);
  });

  // When it looks like we've been disconnected, drop the corresponding
  // connection object from the [connections] global variable.
  chrome.serial.onReceiveError.addListener(function (info) {
    if (info.error == "system_error" || info.error == "disconnected" || info.error == "device_lost")
      connections = connections.filter((x) => x.id != info.connectionId);
  });

  // Probe serial connections at regular intervals. In case we find an mbed port
  // we haven't yet connected to, connect to it.
  setInterval(findNewDevices, 5000);
  findNewDevices();
}

document.addEventListener("DOMContentLoaded", main);