Add face list display

Change-Id: Ide96107acc8ad59cefb0a619968d5c27cd9e94bb
diff --git a/src/face-status.cpp b/src/face-status.cpp
new file mode 100644
index 0000000..5a85ef4
--- /dev/null
+++ b/src/face-status.cpp
@@ -0,0 +1,157 @@
+/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
+/**
+ * Copyright (c) 2013-2017, Regents of the University of California,
+ *
+ * This file is part of NFD Control Center.  See AUTHORS.md for complete list of NFD
+ * authors and contributors.
+ *
+ * NFD Control Center is free software: you can redistribute it and/or modify it under the
+ * terms of the GNU General Public License as published by the Free Software Foundation,
+ * either version 3 of the License, or (at your option) any later version.
+ *
+ * NFD Control Center is distributed in the hope that it will be useful, but WITHOUT ANY
+ * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
+ * PARTICULAR PURPOSE.  See the GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along with NFD
+ * Control Center, e.g., in COPYING.md file.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "face-status.hpp"
+#include "face-status.moc"
+
+namespace ndn {
+
+FaceStatusModel::FaceStatusModel(QObject* parent/* = 0*/)
+  : QAbstractListModel(parent)
+{
+  connect(this, SIGNAL(onDataReceived(std::vector<ndn::nfd::FaceStatus>)), this,
+          SLOT(updateStatus(std::vector<ndn::nfd::FaceStatus>)),
+          Qt::QueuedConnection);
+}
+
+int
+FaceStatusModel::rowCount(const QModelIndex& parent/* = QModelIndex()*/) const
+{
+  return m_items.count();
+}
+
+void
+FaceStatusModel::addItem(const FaceStatusItem& item)
+{
+  beginInsertRows(QModelIndex(), rowCount(), rowCount());
+  m_items << item;
+  endInsertRows();
+}
+
+QVariant
+FaceStatusModel::data(const QModelIndex& index, int role) const
+{
+  if (index.row() < 0 || index.row() >= m_items.count()) {
+    return QVariant();
+  }
+
+  const FaceStatusItem& item = m_items.at(index.row());
+  if (role == FaceIdRole) {
+    return static_cast<uint>(item.faceId());
+  }
+  else if (role == RemoteRole) {
+    return item.remote();
+  }
+  else if (role == LocalRole) {
+    return item.local();
+  }
+  else if (role == ScopeRole) {
+    return item.scope();
+  }
+  else if (role == PersistencyRole) {
+    return item.persistency();
+  }
+  else if (role == LinkTypeRole) {
+    return item.linkType();
+  }
+  else if (role == InInterestRole) {
+    return static_cast<uint>(item.inInterest());
+  }
+  else if (role == OutInterestRole) {
+    return static_cast<uint>(item.outInterest());
+  }
+  else if (role == InDataRole) {
+    return static_cast<uint>(item.inData());
+  }
+  else if (role == OutDataRole) {
+    return static_cast<uint>(item.outData());
+  }
+  else if (role == InByteRole) {
+    return static_cast<uint>(item.inByte());
+  }
+  else if (role == OutByteRole) {
+    return static_cast<uint>(item.outByte());
+  }
+  else if (role == InNackRole) {
+    return static_cast<uint>(item.inNack());
+  }
+  else if (role == OutNackRole) {
+    return static_cast<uint>(item.outNack());
+  }
+
+  return QVariant();
+}
+
+QHash<int, QByteArray>
+FaceStatusModel::roleNames() const
+{
+  QHash<int, QByteArray> roles;
+  roles[FaceIdRole] = "faceId";
+  roles[RemoteRole] = "remote";
+  roles[LocalRole] = "local";
+  roles[ScopeRole] = "scope";
+  roles[PersistencyRole] = "persistency";
+  roles[LinkTypeRole] = "linkType";
+  roles[InInterestRole] = "inInterest";
+  roles[OutInterestRole] = "outInterest";
+  roles[InDataRole] = "inData";
+  roles[OutDataRole] = "outData";
+  roles[InByteRole] = "inByte";
+  roles[OutByteRole] = "outByte";
+  roles[InNackRole] = "inNack";
+  roles[OutNackRole] = "outNack";
+  return roles;
+}
+
+void
+FaceStatusModel::clear()
+{
+  beginResetModel();
+  m_items.clear();
+  endResetModel();
+}
+
+void
+FaceStatusModel::updateStatus(std::vector<ndn::nfd::FaceStatus> status)
+{
+  beginResetModel();
+  m_items.clear();
+  for (auto const& faceEntry : status) {
+    QString faceScope = (faceEntry.getFaceScope() == ndn::nfd::FACE_SCOPE_LOCAL) ? QString("Local") : QString("Non-Local");
+    QString facePersistency = (faceEntry.getFacePersistency() == ndn::nfd::FACE_PERSISTENCY_PERSISTENT) ? QString("Persistent") :
+                              ((faceEntry.getFacePersistency() == ndn::nfd::FACE_PERSISTENCY_PERMANENT) ? QString("Permanent") :
+                              QString("On-Demand"));
+    QString faceLinkType = (faceEntry.getLinkType() == ndn::nfd::LINK_TYPE_MULTI_ACCESS) ? QString("Multi-Access") :
+                           QString("Point-to-Point");
+    addItem(FaceStatusItem(faceEntry.getFaceId(), QString::fromStdString(faceEntry.getRemoteUri()),
+                           QString::fromStdString(faceEntry.getLocalUri()), faceScope, facePersistency, faceLinkType,
+                           faceEntry.getNInInterests(), faceEntry.getNOutInterests(), faceEntry.getNInDatas(),
+                           faceEntry.getNOutDatas(), faceEntry.getNInBytes(), faceEntry.getNOutBytes(),
+                           faceEntry.getNInNacks(), faceEntry.getNOutNacks()));
+  }
+  endResetModel();
+}
+
+void
+FaceStatusModel::onTimeout()
+{
+  std::cerr << "Request timed out (should not really happen)" << std::endl;
+}
+
+} // namespace ndn
diff --git a/src/face-status.hpp b/src/face-status.hpp
new file mode 100644
index 0000000..1131317
--- /dev/null
+++ b/src/face-status.hpp
@@ -0,0 +1,212 @@
+/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
+/**
+ * Copyright (c) 2013-2017, Regents of the University of California,
+ *
+ * This file is part of NFD Control Center.  See AUTHORS.md for complete list of NFD
+ * authors and contributors.
+ *
+ * NFD Control Center is free software: you can redistribute it and/or modify it under the
+ * terms of the GNU General Public License as published by the Free Software Foundation,
+ * either version 3 of the License, or (at your option) any later version.
+ *
+ * NFD Control Center is distributed in the hope that it will be useful, but WITHOUT ANY
+ * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
+ * PARTICULAR PURPOSE.  See the GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along with NFD
+ * Control Center, e.g., in COPYING.md file.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef NCC_FACE_STATUS_HPP
+#define NCC_FACE_STATUS_HPP
+
+#include <QtCore/QAbstractListModel>
+#include <QtCore/QStringList>
+
+#include <ndn-cxx/mgmt/nfd/face-status.hpp>
+
+namespace ndn {
+
+class FaceStatusItem
+{
+public:
+  FaceStatusItem(uint64_t faceId, const QString& remote, const QString& local, const QString& scope, const QString& persistency,
+                 const QString& linkType, uint64_t inInterest, uint64_t outInterest, uint64_t inData, uint64_t outData,
+                 uint64_t inByte, uint64_t outByte, uint64_t inNack, uint64_t outNack)
+    : m_faceId(faceId)
+    , m_remote(remote)
+    , m_local(local)
+    , m_scope(scope)
+    , m_persistency(persistency)
+    , m_linkType(linkType)
+    , m_inInterest(inInterest)
+    , m_outInterest(outInterest)
+    , m_inData(inData)
+    , m_outData(outData)
+    , m_inByte(inByte)
+    , m_outByte(outByte)
+    , m_inNack(inNack)
+    , m_outNack(outNack)
+  {
+  }
+
+  uint64_t
+  faceId() const
+  {
+    return m_faceId;
+  }
+
+  const QString&
+  remote() const
+  {
+    return m_remote;
+  }
+
+  const QString&
+  local() const
+  {
+    return m_local;
+  }
+
+  const QString&
+  scope() const
+  {
+    return m_scope;
+  }
+
+  const QString&
+  persistency() const
+  {
+    return m_persistency;
+  }
+
+  const QString&
+  linkType() const
+  {
+    return m_linkType;
+  }
+
+  uint64_t
+  inInterest() const
+  {
+    return m_inInterest;
+  }
+
+  uint64_t
+  outInterest() const
+  {
+    return m_outInterest;
+  }
+
+  uint64_t
+  inData() const
+  {
+    return m_inData;
+  }
+
+  uint64_t
+  outData() const
+  {
+    return m_outData;
+  }
+
+  uint64_t
+  inByte() const
+  {
+    return m_inByte;
+  }
+
+  uint64_t
+  outByte() const
+  {
+    return m_outByte;
+  }
+
+  uint64_t
+  inNack() const
+  {
+    return m_inNack;
+  }
+
+  uint64_t
+  outNack() const
+  {
+    return m_outNack;
+  }
+
+private:
+  uint64_t m_faceId;
+  QString m_remote;
+  QString m_local;
+  QString m_scope;
+  QString m_persistency;
+  QString m_linkType;
+  uint64_t m_inInterest;
+  uint64_t m_outInterest;
+  uint64_t m_inData;
+  uint64_t m_outData;
+  uint64_t m_inByte;
+  uint64_t m_outByte;
+  uint64_t m_inNack;
+  uint64_t m_outNack;
+};
+
+class FaceStatusModel : public QAbstractListModel
+{
+  Q_OBJECT
+
+signals:
+  void
+  onDataReceived(std::vector<ndn::nfd::FaceStatus> status);
+
+public:
+  enum FaceStatusRoles {
+    FaceIdRole = Qt::UserRole + 1,
+    RemoteRole,
+    LocalRole,
+    ScopeRole,
+    PersistencyRole,
+    LinkTypeRole,
+    InInterestRole,
+    OutInterestRole,
+    InDataRole,
+    OutDataRole,
+    InByteRole,
+    OutByteRole,
+    InNackRole,
+    OutNackRole
+  };
+
+  explicit
+  FaceStatusModel(QObject* parent = 0);
+
+  int
+  rowCount(const QModelIndex& parent = QModelIndex()) const;
+
+  void
+  addItem(const FaceStatusItem& item);
+
+  QVariant
+  data(const QModelIndex& index, int role) const;
+
+  QHash<int, QByteArray>
+  roleNames() const;
+
+  void
+  clear();
+
+  void
+  onTimeout();
+
+private slots:
+
+  void
+  updateStatus(std::vector<ndn::nfd::FaceStatus> status);
+
+private:
+  QList<FaceStatusItem> m_items;
+};
+
+} // namespace ndn
+
+#endif // NCC_FACE_STATUS_HPP
diff --git a/src/main.cpp b/src/main.cpp
index a3e125f..9d42157 100644
--- a/src/main.cpp
+++ b/src/main.cpp
@@ -160,6 +160,7 @@
 Q_DECLARE_METATYPE(ndn::nfd::ForwarderStatus)
 Q_DECLARE_METATYPE(std::vector<ndn::nfd::FibEntry>)
 Q_DECLARE_METATYPE(std::vector<ndn::nfd::RibEntry>)
+Q_DECLARE_METATYPE(std::vector<ndn::nfd::FaceStatus>)
 
 int
 main(int argc, char *argv[])
@@ -168,6 +169,7 @@
   qRegisterMetaType<ndn::nfd::ForwarderStatus>();
   qRegisterMetaType<std::vector<ndn::nfd::FibEntry>>();
   qRegisterMetaType<std::vector<ndn::nfd::RibEntry>>();
+  qRegisterMetaType<std::vector<ndn::nfd::FaceStatus>>();
 
   QApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
   QApplication app(argc, argv);
diff --git a/src/status-viewer.cpp b/src/status-viewer.cpp
index de7bab9..6bc5879 100644
--- a/src/status-viewer.cpp
+++ b/src/status-viewer.cpp
@@ -31,6 +31,7 @@
   QQmlContext* s_context = s_engine.rootContext();
 
   s_context->setContextProperty("forwarderModel", &s_forwarderStatusModel);
+  s_context->setContextProperty("faceModel", &s_faceModel);
   s_context->setContextProperty("fibModel", &s_fibModel);
   s_context->setContextProperty("ribModel", &s_ribModel);
   s_context->setContextProperty("statusViewer", this);
@@ -45,6 +46,12 @@
 }
 
 void
+StatusViewer::onFaceStatusRetrieved(const std::vector<nfd::FaceStatus>& status)
+{
+  emit s_faceModel.onDataReceived(status);
+}
+
+void
 StatusViewer::onFibStatusRetrieved(const std::vector<nfd::FibEntry>& status)
 {
   emit s_fibModel.onDataReceived(status);
@@ -68,6 +75,8 @@
 {
   s_controller->fetch<ndn::nfd::ForwarderGeneralStatusDataset>(bind(&StatusViewer::onStatusRetrieved, this, _1),
                                                               bind(&StatusViewer::onStatusTimeout, this));
+  s_controller->fetch<ndn::nfd::FaceDataset>(bind(&StatusViewer::onFaceStatusRetrieved, this, _1),
+                                             bind(&StatusViewer::onStatusTimeout, this));
   s_controller->fetch<ndn::nfd::FibDataset>(bind(&StatusViewer::onFibStatusRetrieved, this, _1),
                                            bind(&StatusViewer::onStatusTimeout, this));
   s_controller->fetch<ndn::nfd::RibDataset>(bind(&StatusViewer::onRibStatusRetrieved, this, _1),
diff --git a/src/status-viewer.hpp b/src/status-viewer.hpp
index e7e4113..b627bdb 100644
--- a/src/status-viewer.hpp
+++ b/src/status-viewer.hpp
@@ -21,12 +21,14 @@
 #define NCC_STATUS_VIEWER_HPP
 
 #include "forwarder-status.hpp"
+#include "face-status.hpp"
 #include "fib-status.hpp"
 #include "rib-status.hpp"
 
 #include <ndn-cxx/face.hpp>
 #include <ndn-cxx/name.hpp>
 #include <ndn-cxx/util/scheduler.hpp>
+#include <ndn-cxx/mgmt/nfd/face-status.hpp>
 #include <ndn-cxx/mgmt/nfd/fib-entry.hpp>
 #include <ndn-cxx/mgmt/nfd/rib-entry.hpp>
 #include <ndn-cxx/mgmt/nfd/controller.hpp>
@@ -49,6 +51,9 @@
   onStatusRetrieved(const nfd::ForwarderStatus& status);
 
   void
+  onFaceStatusRetrieved(const std::vector<nfd::FaceStatus>& status);
+
+  void
   onFibStatusRetrieved(const std::vector<nfd::FibEntry>& status);
 
   void
@@ -78,6 +83,7 @@
   QQmlApplicationEngine s_engine;
 
   ForwarderStatusModel s_forwarderStatusModel;
+  FaceStatusModel s_faceModel;
   FibStatusModel s_fibModel;
   RibStatusModel s_ribModel;
 };
diff --git a/src/status.qml b/src/status.qml
index c288056..c4bb56c 100644
--- a/src/status.qml
+++ b/src/status.qml
@@ -39,6 +39,87 @@
             }
         }
         Tab {
+            title: "Face"
+            TableView {
+                anchors.fill: parent
+                anchors.topMargin: 20
+                anchors.bottomMargin: 20
+                anchors.leftMargin: 20
+                anchors.rightMargin: 20
+                model: faceModel
+                TableViewColumn{
+                    role: "faceId"
+                    title: "Face ID"
+                    width: 60
+                }
+                TableViewColumn{
+                    role: "remote"
+                    title: "Remote URI"
+                    width: 150
+                }
+                TableViewColumn{
+                    role: "local"
+                    title: "Local URI"
+                    width: 150
+                }
+                TableViewColumn{
+                    role: "scope"
+                    title: "Scope"
+                    width: 70
+                }
+                TableViewColumn{
+                    role: "persistency"
+                    title: "Persistency"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "linkType"
+                    title: "LinkType"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "inInterest"
+                    title: "In Interests"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "outInterest"
+                    title: "Out Interests"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "inData"
+                    title: "In Data"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "outData"
+                    title: "Out Data"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "inByte"
+                    title: "In Bytes"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "outByte"
+                    title: "Out Bytes"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "inNack"
+                    title: "In Nacks"
+                    width: 100
+                }
+                TableViewColumn{
+                    role: "outNack"
+                    title: "Out Nacks"
+                    width: 100
+                }
+            }
+        }
+        Tab {
             title: "FIB"
             TableView {
                 anchors.fill: parent
diff --git a/wscript b/wscript
index 38c74cc..0b9396e 100644
--- a/wscript
+++ b/wscript
@@ -42,7 +42,7 @@
         features=['qt5', 'cxxprogram', 'cxx'],
         includes = ". src",
         use = "NDN_CXX BOOST QT5CORE QT5DBUS QT5QML QT5WIDGETS",
-        moc = "src/tray-menu.hpp src/key-tree-model.hpp src/key-viewer-dialog.hpp src/cert-tree-model.hpp src/status-viewer.hpp src/fib-status.hpp src/rib-status.hpp src/forwarder-status.hpp",
+        moc = "src/tray-menu.hpp src/key-tree-model.hpp src/key-viewer-dialog.hpp src/cert-tree-model.hpp src/status-viewer.hpp src/face-status.hpp src/fib-status.hpp src/rib-status.hpp src/forwarder-status.hpp",
         source = bld.path.ant_glob(['src/*.cpp', 'src/**/*.qrc', 'src/**/*.ui', 'src/**/*.qrc'], excl=['src/osx-*']),
         )