enable history
diff --git a/chatdialog.cpp b/chatdialog.cpp
index ba74fb7..1b0329c 100644
--- a/chatdialog.cpp
+++ b/chatdialog.cpp
@@ -8,6 +8,7 @@
 #include <QMessageBox>
 #include <boost/random/random_device.hpp>
 #include <boost/random/uniform_int_distribution.hpp>
+#include <boost/lexical_cast.hpp>
 #include <stdio.h>
 
 #define BROADCAST_PREFIX_FOR_SYNC_DEMO "/ndn/broadcast/chronos"
@@ -18,7 +19,7 @@
 static const int HELLO_INTERVAL = 90;  // seconds
 
 ChatDialog::ChatDialog(QWidget *parent)
-  : QDialog(parent), m_sock(NULL), m_lastMsgTime(0)
+  : QDialog(parent), m_sock(NULL), m_lastMsgTime(0), m_historyInitialized(false)
 {
   // have to register this, otherwise
   // the signal-slot system won't recognize this type
@@ -73,6 +74,8 @@
       m_sock = new Sync::SyncAppSocket(syncPrefix,
                                        bind(&ChatDialog::processTreeUpdateWrapper, this, _1, _2),
                                        bind(&ChatDialog::processRemoveWrapper, this, _1));
+      Sync::CcnxWrapperPtr handle = Sync::CcnxWrapper::Create();
+      handle->setInterestFilter(m_user.getPrefix().toStdString(), bind(&ChatDialog::respondHistoryRequest, this, _1));
       QTimer::singleShot(600, this, SLOT(sendJoin()));
       m_timer->start(FRESHNESS * 2000);
       disableTreeDisplay();
@@ -205,6 +208,13 @@
       return;
     }
 
+    if (m_history.size() == MAX_HISTORY_ENTRY)
+    {
+      m_history.dequeue();
+    }
+
+    m_history.enqueue(msg);
+
     QTextCharFormat nickFormat;
     nickFormat.setForeground(Qt::darkGreen);
     nickFormat.setFontWeight(QFont::Bold);
@@ -377,6 +387,44 @@
   char *tempBuf = new char[len];
   memcpy(tempBuf, buf, len);
   emit dataReceived(name.c_str(), tempBuf, len, false);
+  
+  if (!m_historyInitialized)
+  {
+    fetchHistory(name);
+    m_historyInitialized = true;
+  }
+}
+
+void 
+ChatDialog::fetchHistory(std::string name)
+{
+  std::string nameWithoutSeq = name.substr(0, name.find_last_of('/'));
+  std::string prefix = nameWithoutSeq.substr(0, nameWithoutSeq.find_last_of('/'));
+  prefix += "/history";
+  Sync::CcnxWrapperPtr handle = Sync::CcnxWrapper::Create(); 
+  QString randomString = getRandomString();
+  for (int i = 0; i < MAX_HISTORY_ENTRY; i++)
+  {
+    QString interest = QString("%1/%2/%3").arg(prefix.c_str()).arg(randomString).arg(i);
+    handle->sendInterest(interest.toStdString(), bind(&ChatDialog::processDataWrapper, this, _1, _2, _3));
+  }
+}
+
+void
+ChatDialog::respondHistoryRequest(std::string interest)
+{
+  std::string seqStr = interest.substr(interest.find_last_of('/') + 1); 
+  int seq = boost::lexical_cast<int>(seqStr);
+  if (seq >= 0 && seq < m_history.size())
+  {
+    Sync::CcnxWrapperPtr handle = Sync::CcnxWrapper::Create();
+    SyncDemo::ChatMessage msg = m_history.at(seq);
+    size_t size = msg.ByteSize();
+    char *buf = new char[size];
+    msg.SerializeToArray(buf, size);
+    handle->publishRawData(interest, buf, size, 1);
+    delete buf;
+  }
 }
 
 void
@@ -707,6 +755,7 @@
     m_user.setNick(nick);
     needWrite = true;
   }
+  QString oldPrefix = m_user.getPrefix();
   if (!originPrefix.isEmpty() && originPrefix != m_user.getOriginPrefix()) {
     m_user.setOriginPrefix(originPrefix);
     m_user.setPrefix(originPrefix + "/" + m_user.getChatroom() + "/" + randString);
@@ -735,6 +784,10 @@
     if (m_sock != NULL) 
     {
       sendLeave();
+      Sync::CcnxWrapperPtr handle = Sync::CcnxWrapper::Create();
+      handle->clearInterestFilter(oldPrefix.toStdString());
+      m_history.clear();
+      m_historyInitialized = false;
       delete m_sock;
       m_sock = NULL;
     }
@@ -744,6 +797,8 @@
     try
     {
       m_sock = new Sync::SyncAppSocket(syncPrefix, bind(&ChatDialog::processTreeUpdateWrapper, this, _1, _2), bind(&ChatDialog::processRemoveWrapper, this, _1));
+      Sync::CcnxWrapperPtr handle = Sync::CcnxWrapper::Create();
+      handle->setInterestFilter(m_user.getPrefix().toStdString(), bind(&ChatDialog::respondHistoryRequest, this, _1));
       QTimer::singleShot(1000, this, SLOT(sendJoin()));
       m_timer->start(FRESHNESS * 2000);
       disableTreeDisplay();
diff --git a/chatdialog.h b/chatdialog.h
index 840afb2..5319f37 100644
--- a/chatdialog.h
+++ b/chatdialog.h
@@ -10,9 +10,11 @@
 #include <sync-logic.h>
 #include <sync-seq-no.h>
 #include <QSystemTrayIcon>
+#include <QQueue>
 
 #define ORGANIZATION "IRL@UCLA"
 #define APPLICATION "CHRONOS"
+#define MAX_HISTORY_ENTRY   20
 
 class QAction;
 class QMenu;
@@ -32,6 +34,7 @@
   void processDataWrapper(std::string, const char *buf, size_t len);
   void processDataNoShowWrapper(std::string, const char *buf, size_t len);
   void processRemoveWrapper(std::string);
+  void respondHistoryRequest(std::string interest);
 
 protected:
   void closeEvent(QCloseEvent *e);
@@ -43,6 +46,7 @@
   void processRemove(QString);
 
 private:
+  void fetchHistory(std::string name);
   QString getRandomString();
   void formChatMessage(const QString &text, SyncDemo::ChatMessage &msg);
   void formControlMessage(SyncDemo::ChatMessage &msg, SyncDemo::ChatMessage::ChatMessageType type);
@@ -97,6 +101,9 @@
   QStringListModel *m_rosterModel;
   bool m_minimaniho;
 
+  QQueue<SyncDemo::ChatMessage> m_history;
+  bool m_historyInitialized;
+  
   // icon related
   QAction *minimizeAction;
   QAction *maximizeAction;