blob: 313c67390008608143d0efccc6460aff3c739894 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
/*
* Copyright (c) 2013, Regents of the University of California
* Yingdi Yu
*
* BSD license, See the LICENSE file for more information
*
* Author: Yingdi Yu <yingdi@cs.ucla.edu>
*/
#include <QApplication>
#include <QMessageBox>
#include <QDir>
#include <QTimer>
#include "controller.hpp"
//#include "chatroom-discovery.h"
#ifndef Q_MOC_RUN
#include <boost/filesystem.hpp>
#include <boost/lexical_cast.hpp>
#include <ndn-cxx/util/random.hpp>
#include "cryptopp.hpp"
#include "config.pb.h"
#include "endorse-info.pb.h"
#include "logging.h"
#endif
// INIT_LOGGER("chronos.Controller");
Q_DECLARE_METATYPE(ndn::Name)
Q_DECLARE_METATYPE(ndn::IdentityCertificate)
Q_DECLARE_METATYPE(Chronos::EndorseInfo)
Q_DECLARE_METATYPE(ndn::Interest)
Q_DECLARE_METATYPE(size_t)
Q_DECLARE_METATYPE(chronos::ChatroomInfo)
namespace chronos {
using std::string;
using ndn::Face;
using ndn::IdentityCertificate;
using ndn::OnInterestValidated;
using ndn::OnInterestValidationFailed;
static const uint8_t ROUTING_PREFIX_SEPARATOR[2] = {0xF0, 0x2E};
// constructor & destructor
Controller::Controller(shared_ptr<Face> face,
QWidget* parent)
: QDialog(parent)
, m_face(face)
, m_invitationListenerId(0)
, m_contactManager(m_face)
, m_discoveryLogic(m_face,
bind(&Controller::updateDiscoveryList, this, _1, _2))
, m_settingDialog(new SettingDialog)
, m_startChatDialog(new StartChatDialog)
, m_profileEditor(new ProfileEditor)
, m_invitationDialog(new InvitationDialog)
, m_contactPanel(new ContactPanel)
, m_browseContactDialog(new BrowseContactDialog)
, m_addContactPanel(new AddContactPanel)
, m_chatroomDiscoveryDialog(new ChatroomDiscoveryDialog)
{
qRegisterMetaType<ndn::Name>("ndn.Name");
qRegisterMetaType<ndn::IdentityCertificate>("ndn.IdentityCertificate");
qRegisterMetaType<Chronos::EndorseInfo>("Chronos.EndorseInfo");
qRegisterMetaType<ndn::Interest>("ndn.Interest");
qRegisterMetaType<size_t>("size_t");
qRegisterMetaType<chronos::ChatroomInfo>("chronos.Chatroom");
connect(this, SIGNAL(localPrefixUpdated(const QString&)),
this, SLOT(onLocalPrefixUpdated(const QString&)));
connect(this, SIGNAL(invitationInterest(const ndn::Name&, const ndn::Interest&, size_t)),
this, SLOT(onInvitationInterest(const ndn::Name&, const ndn::Interest&, size_t)));
// Connection to ContactManager
connect(this, SIGNAL(identityUpdated(const QString&)),
&m_contactManager, SLOT(onIdentityUpdated(const QString&)));
connect(&m_contactManager, SIGNAL(warning(const QString&)),
this, SLOT(onWarning(const QString&)));
connect(this, SIGNAL(refreshBrowseContact()),
&m_contactManager, SLOT(onRefreshBrowseContact()));
connect(&m_contactManager, SIGNAL(contactInfoFetchFailed(const QString&)),
this, SLOT(onWarning(const QString&)));
connect(&m_contactManager, SIGNAL(contactIdListReady(const QStringList&)),
this, SLOT(onContactIdListReady(const QStringList&)));
// Connection to SettingDialog
connect(this, SIGNAL(identityUpdated(const QString&)),
m_settingDialog, SLOT(onIdentityUpdated(const QString&)));
connect(m_settingDialog, SIGNAL(identityUpdated(const QString&)),
this, SLOT(onIdentityUpdated(const QString&)));
connect(m_settingDialog, SIGNAL(nickUpdated(const QString&)),
this, SLOT(onNickUpdated(const QString&)));
// Connection to ProfileEditor
connect(this, SIGNAL(closeDBModule()),
m_profileEditor, SLOT(onCloseDBModule()));
connect(this, SIGNAL(identityUpdated(const QString&)),
m_profileEditor, SLOT(onIdentityUpdated(const QString&)));
connect(m_profileEditor, SIGNAL(updateProfile()),
&m_contactManager, SLOT(onUpdateProfile()));
// Connection to StartChatDialog
connect(m_startChatDialog, SIGNAL(startChatroom(const QString&, bool)),
this, SLOT(onStartChatroom(const QString&, bool)));
// Connection to InvitationDialog
connect(m_invitationDialog, SIGNAL(invitationResponded(const ndn::Name&, bool)),
this, SLOT(onInvitationResponded(const ndn::Name&, bool)));
// Connection to AddContactPanel
connect(m_addContactPanel, SIGNAL(fetchInfo(const QString&)),
&m_contactManager, SLOT(onFetchContactInfo(const QString&)));
connect(m_addContactPanel, SIGNAL(addContact(const QString&)),
&m_contactManager, SLOT(onAddFetchedContact(const QString&)));
connect(&m_contactManager, SIGNAL(contactEndorseInfoReady(const Chronos::EndorseInfo&)),
m_addContactPanel, SLOT(onContactEndorseInfoReady(const Chronos::EndorseInfo&)));
// Connection to BrowseContactDialog
connect(m_browseContactDialog, SIGNAL(directAddClicked()),
this, SLOT(onDirectAdd()));
connect(m_browseContactDialog, SIGNAL(fetchIdCert(const QString&)),
&m_contactManager, SLOT(onFetchIdCert(const QString&)));
connect(m_browseContactDialog, SIGNAL(addContact(const QString&)),
&m_contactManager, SLOT(onAddFetchedContactIdCert(const QString&)));
connect(&m_contactManager, SIGNAL(idCertNameListReady(const QStringList&)),
m_browseContactDialog, SLOT(onIdCertNameListReady(const QStringList&)));
connect(&m_contactManager, SIGNAL(nameListReady(const QStringList&)),
m_browseContactDialog, SLOT(onNameListReady(const QStringList&)));
connect(&m_contactManager, SIGNAL(idCertReady(const ndn::IdentityCertificate&)),
m_browseContactDialog, SLOT(onIdCertReady(const ndn::IdentityCertificate&)));
// Connection to ContactPanel
connect(m_contactPanel, SIGNAL(waitForContactList()),
&m_contactManager, SLOT(onWaitForContactList()));
connect(m_contactPanel, SIGNAL(waitForContactInfo(const QString&)),
&m_contactManager, SLOT(onWaitForContactInfo(const QString&)));
connect(m_contactPanel, SIGNAL(removeContact(const QString&)),
&m_contactManager, SLOT(onRemoveContact(const QString&)));
connect(m_contactPanel, SIGNAL(updateAlias(const QString&, const QString&)),
&m_contactManager, SLOT(onUpdateAlias(const QString&, const QString&)));
connect(m_contactPanel, SIGNAL(updateIsIntroducer(const QString&, bool)),
&m_contactManager, SLOT(onUpdateIsIntroducer(const QString&, bool)));
connect(m_contactPanel, SIGNAL(updateEndorseCertificate(const QString&)),
&m_contactManager, SLOT(onUpdateEndorseCertificate(const QString&)));
connect(m_contactPanel, SIGNAL(warning(const QString&)),
this, SLOT(onWarning(const QString&)));
connect(this, SIGNAL(closeDBModule()),
m_contactPanel, SLOT(onCloseDBModule()));
connect(this, SIGNAL(identityUpdated(const QString&)),
m_contactPanel, SLOT(onIdentityUpdated(const QString&)));
connect(&m_contactManager, SIGNAL(contactAliasListReady(const QStringList&)),
m_contactPanel, SLOT(onContactAliasListReady(const QStringList&)));
connect(&m_contactManager, SIGNAL(contactIdListReady(const QStringList&)),
m_contactPanel, SLOT(onContactIdListReady(const QStringList&)));
connect(&m_contactManager, SIGNAL(contactInfoReady(const QString&, const QString&,
const QString&, bool)),
m_contactPanel, SLOT(onContactInfoReady(const QString&, const QString&,
const QString&, bool)));
// Connection to DiscoveryDialog
connect(this,
SIGNAL(discoverChatroomChanged(const chronos::ChatroomInfo&, bool)),
m_chatroomDiscoveryDialog,
SLOT(onDiscoverChatroomChanged(const chronos::ChatroomInfo&, bool)));
connect(m_chatroomDiscoveryDialog, SIGNAL(startChatroom(const QString&, bool)),
this, SLOT(onStartChatroom(const QString&, bool)));
initialize();
createTrayIcon();
onUpdateLocalPrefixAction();
}
Controller::~Controller()
{
saveConf();
}
// public methods
// private methods
string
Controller::getDBName()
{
string dbName("chronos-");
std::stringstream ss;
{
using namespace CryptoPP;
SHA256 hash;
StringSource(m_identity.wireEncode().wire(), m_identity.wireEncode().size(), true,
new HashFilter(hash, new HexEncoder(new FileSink(ss), false)));
}
dbName.append(ss.str()).append(".db");
return dbName;
}
void
Controller::openDB()
{
m_db = QSqlDatabase::addDatabase("QSQLITE");
QString path = (QDir::home().path());
path.append(QDir::separator())
.append(".chronos")
.append(QDir::separator())
.append(getDBName().c_str());
m_db.setDatabaseName(path);
m_db.open();
// bool ok = m_db.open();
// _LOG_DEBUG("DB opened: " << std::boolalpha << ok );
}
void
Controller::initialize()
{
loadConf();
m_keyChain.createIdentity(m_identity);
openDB();
emit identityUpdated(QString(m_identity.toUri().c_str()));
setInvitationListener();
m_discoveryLogic.sendDiscoveryInterest();
}
void
Controller::setInvitationListener()
{
if (m_invitationListenerId != 0)
m_face->unsetInterestFilter(m_invitationListenerId);
Name invitationPrefix;
Name routingPrefix = getInvitationRoutingPrefix();
size_t offset = 0;
if (!routingPrefix.isPrefixOf(m_identity)) {
invitationPrefix.append(routingPrefix).append(ROUTING_PREFIX_SEPARATOR, 2);
offset = routingPrefix.size() + 1;
}
invitationPrefix.append(m_identity).append("CHRONOCHAT-INVITATION");
m_invitationListenerId = m_face->setInterestFilter(invitationPrefix,
bind(&Controller::onInvitationInterestWrapper,
this, _1, _2, offset),
bind(&Controller::onInvitationRegisterFailed,
this, _1, _2));
}
void
Controller::loadConf()
{
namespace fs = boost::filesystem;
fs::path chronosDir = fs::path(getenv("HOME")) / ".chronos";
fs::create_directories (chronosDir);
std::ifstream is((chronosDir / "config").c_str ());
ChronoChat::Conf conf;
if (conf.ParseFromIstream(&is)) {
m_identity.clear();
m_identity.append(conf.identity());
if (conf.has_nick())
m_nick = conf.nick();
else
m_nick = m_identity.get(-1).toUri();
}
else {
m_identity.clear();
// TODO: change below to system default;
m_identity.append("chronochat-tmp-identity")
.append(getRandomString());
m_nick = m_identity.get(-1).toUri();
}
}
void
Controller::saveConf()
{
namespace fs = boost::filesystem;
fs::path chronosDir = fs::path(getenv("HOME")) / ".chronos";
fs::create_directories (chronosDir);
std::ofstream os((chronosDir / "config").c_str ());
ChronoChat::Conf conf;
conf.set_identity(m_identity.toUri());
if (!m_nick.empty())
conf.set_nick(m_nick);
conf.SerializeToOstream(&os);
os.close();
}
void
Controller::createActions()
{
m_startChatroom = new QAction(tr("Start new chat"), this);
connect(m_startChatroom, SIGNAL(triggered()), this, SLOT(onStartChatAction()));
m_discoveryAction = new QAction(tr("Ongoing Chatrooms"), this);
connect(m_discoveryAction, SIGNAL(triggered()), this, SLOT(onDiscoveryAction()));
m_settingsAction = new QAction(tr("Settings"), this);
connect(m_settingsAction, SIGNAL(triggered()), this, SLOT(onSettingsAction()));
m_editProfileAction = new QAction(tr("Edit profile"), this);
connect(m_editProfileAction, SIGNAL(triggered()), this, SLOT(onProfileEditorAction()));
m_contactListAction = new QAction(tr("Contact List"), this);
connect(m_contactListAction, SIGNAL(triggered()), this, SLOT(onContactListAction()));
m_addContactAction = new QAction(tr("Add contact"), this);
connect(m_addContactAction, SIGNAL(triggered()), this, SLOT(onAddContactAction()));
m_updateLocalPrefixAction = new QAction(tr("Update local prefix"), this);
connect(m_updateLocalPrefixAction, SIGNAL(triggered()), this, SLOT(onUpdateLocalPrefixAction()));
m_minimizeAction = new QAction(tr("Mi&nimize"), this);
connect(m_minimizeAction, SIGNAL(triggered()), this, SLOT(onMinimizeAction()));
m_quitAction = new QAction(tr("Quit"), this);
connect(m_quitAction, SIGNAL(triggered()), this, SLOT(onQuitAction()));
}
void
Controller::createTrayIcon()
{
createActions();
m_trayIconMenu = new QMenu(this);
m_trayIconMenu->addAction(m_startChatroom);
m_trayIconMenu->addAction(m_discoveryAction);
m_trayIconMenu->addSeparator();
m_trayIconMenu->addAction(m_settingsAction);
m_trayIconMenu->addAction(m_editProfileAction);
m_trayIconMenu->addSeparator();
m_trayIconMenu->addAction(m_contactListAction);
m_trayIconMenu->addAction(m_addContactAction);
m_trayIconMenu->addSeparator();
m_trayIconMenu->addAction(m_updateLocalPrefixAction);
m_trayIconMenu->addSeparator();
m_trayIconMenu->addAction(m_minimizeAction);
m_closeMenu = m_trayIconMenu->addMenu("Close chatroom");
m_closeMenu->setEnabled(false);
m_trayIconMenu->addSeparator();
m_trayIconMenu->addAction(m_quitAction);
m_trayIcon = new QSystemTrayIcon(this);
m_trayIcon->setContextMenu(m_trayIconMenu);
m_trayIcon->setIcon(QIcon(":/images/icon_small.png"));
m_trayIcon->setToolTip("ChronoChat System Tray Icon");
m_trayIcon->setVisible(true);
}
void
Controller::updateMenu()
{
QMenu* menu = new QMenu(this);
QMenu* closeMenu = 0;
menu->addAction(m_startChatroom);
menu->addAction(m_discoveryAction);
menu->addSeparator();
menu->addAction(m_settingsAction);
menu->addAction(m_editProfileAction);
menu->addSeparator();
menu->addAction(m_contactListAction);
menu->addAction(m_addContactAction);
menu->addSeparator();
{
ChatActionList::const_iterator it = m_chatActionList.begin();
ChatActionList::const_iterator end = m_chatActionList.end();
if (it != end) {
for (; it != end; it++)
menu->addAction(it->second);
menu->addSeparator();
}
}
menu->addAction(m_updateLocalPrefixAction);
menu->addSeparator();
menu->addAction(m_minimizeAction);
closeMenu = menu->addMenu("Close chatroom");
{
ChatActionList::const_iterator it = m_closeActionList.begin();
ChatActionList::const_iterator end = m_closeActionList.end();
if (it == end)
closeMenu->setEnabled(false);
else
for (; it != end; it++)
closeMenu->addAction(it->second);
}
menu->addSeparator();
menu->addAction(m_quitAction);
m_trayIcon->setContextMenu(menu);
delete m_trayIconMenu;
m_trayIconMenu = menu;
m_closeMenu = closeMenu;
}
void
Controller::onLocalPrefix(const Interest& interest, Data& data)
{
QString localPrefixStr("/private/local");
Name prefix;
Block contentBlock = data.getContent();
try {
contentBlock.parse();
for (Block::element_const_iterator it = contentBlock.elements_begin();
it != contentBlock.elements_end(); it++) {
Name candidate;
candidate.wireDecode(*it);
if (candidate.isPrefixOf(m_identity)) {
prefix = candidate;
break;
}
}
if (prefix.empty()) {
if (contentBlock.elements_begin() != contentBlock.elements_end())
prefix.wireDecode(*contentBlock.elements_begin());
else
prefix = Name("/private/local");
}
localPrefixStr = QString::fromStdString(prefix.toUri());
}
catch (Block::Error& e) {
prefix = Name("/private/local");
}
if (m_localPrefix.empty() || m_localPrefix != prefix) {
emit localPrefixUpdated(localPrefixStr);
}
}
void
Controller::onLocalPrefixTimeout(const Interest& interest)
{
QString localPrefixStr("/private/local");
Name localPrefix(localPrefixStr.toStdString());
if (m_localPrefix.empty() || m_localPrefix != localPrefix) {
emit localPrefixUpdated(localPrefixStr);
}
}
void
Controller::onInvitationInterestWrapper(const Name& prefix,
const Interest& interest,
size_t routingPrefixOffset)
{
emit invitationInterest(prefix, interest, routingPrefixOffset);
}
void
Controller::onInvitationRegisterFailed(const Name& prefix, const string& failInfo)
{
// _LOG_DEBUG("Controller::onInvitationRegisterFailed: " << failInfo);
}
void
Controller::onInvitationValidated(const shared_ptr<const Interest>& interest)
{
Invitation invitation(interest->getName());
// Should be obtained via a method of ContactManager.
string alias = invitation.getInviterCertificate().getPublicKeyName().getPrefix(-1).toUri();
m_invitationDialog->setInvitation(alias, invitation.getChatroom(), interest->getName());
m_invitationDialog->show();
}
void
Controller::onInvitationValidationFailed(const shared_ptr<const Interest>& interest,
string failureInfo)
{
// _LOG_DEBUG("Invitation: " << interest->getName() <<
// " cannot not be validated due to: " << failureInfo);
}
string
Controller::getRandomString()
{
uint32_t r = ndn::random::generateWord32();
std::stringstream ss;
{
using namespace CryptoPP;
StringSource(reinterpret_cast<uint8_t*>(&r), 4, true,
new HexEncoder(new FileSink(ss), false));
}
// for (int i = 0; i < 8; i++)
// {
// uint32_t t = r & mask;
// if (t < 10)
// ss << static_cast<char>(t + 0x30);
// else
// ss << static_cast<char>(t + 0x57);
// r = r >> 4;
// }
return ss.str();
}
ndn::Name
Controller::getInvitationRoutingPrefix()
{
return Name("/ndn/broadcast");
}
void
Controller::addChatDialog(const QString& chatroomName, ChatDialog* chatDialog)
{
m_chatDialogList[chatroomName.toStdString()] = chatDialog;
m_discoveryLogic.addLocalChatroom(*chatDialog->getChatroomInfo());
connect(chatDialog, SIGNAL(closeChatDialog(const QString&)),
this, SLOT(onRemoveChatDialog(const QString&)));
connect(chatDialog, SIGNAL(showChatMessage(const QString&, const QString&, const QString&)),
this, SLOT(onShowChatMessage(const QString&, const QString&, const QString&)));
connect(chatDialog, SIGNAL(resetIcon()),
this, SLOT(onResetIcon()));
connect(chatDialog, SIGNAL(rosterChanged(const chronos::ChatroomInfo&)),
this, SLOT(onRosterChanged(const chronos::ChatroomInfo&)));
connect(this, SIGNAL(localPrefixUpdated(const QString&)),
chatDialog, SLOT(onLocalPrefixUpdated(const QString&)));
QAction* chatAction = new QAction(chatroomName, this);
m_chatActionList[chatroomName.toStdString()] = chatAction;
connect(chatAction, SIGNAL(triggered()),
chatDialog, SLOT(raise()));//ymj
QAction* closeAction = new QAction(chatroomName, this);
m_closeActionList[chatroomName.toStdString()] = closeAction;
connect(closeAction, SIGNAL(triggered()), chatDialog, SLOT(onClose()));
updateMenu();
}
void
Controller::updateDiscoveryList(const ChatroomInfo& info, bool isAdd)
{
emit discoverChatroomChanged(info, isAdd);
}
void
Controller::onIdentityUpdated(const QString& identity)
{
Name identityName(identity.toStdString());
while (!m_chatDialogList.empty()) {
ChatDialogList::const_iterator it = m_chatDialogList.begin();
onRemoveChatDialog(QString::fromStdString(it->first));
}
m_identity = identityName;
m_keyChain.createIdentity(m_identity);
setInvitationListener();
emit closeDBModule();
QTimer::singleShot(500, this, SLOT(onIdentityUpdatedContinued()));
}
void
Controller::onIdentityUpdatedContinued()
{
QString connection = m_db.connectionName();
// _LOG_DEBUG("connection name: " << connection.toStdString());
QSqlDatabase::removeDatabase(connection);
m_db.close();
openDB();
emit identityUpdated(QString(m_identity.toUri().c_str()));
}
void
Controller::onContactIdListReady(const QStringList& list)
{
ContactList contactList;
m_contactManager.getContactList(contactList);
m_validator.cleanTrustAnchor();
for (ContactList::const_iterator it = contactList.begin(); it != contactList.end(); it++)
m_validator.addTrustAnchor((*it)->getPublicKeyName(), (*it)->getPublicKey());
}
void
Controller::onNickUpdated(const QString& nick)
{
m_nick = nick.toStdString();
}
void
Controller::onLocalPrefixUpdated(const QString& localPrefix)
{
m_localPrefix = Name(localPrefix.toStdString());
}
void
Controller::onStartChatAction()
{
string chatroom = "chatroom-" + getRandomString();
m_startChatDialog->setChatroom(chatroom);
m_startChatDialog->show();
m_startChatDialog->raise();
}
void
Controller::onDiscoveryAction()
{
m_discoveryLogic.sendDiscoveryInterest();
m_chatroomDiscoveryDialog->updateChatroomList();
m_chatroomDiscoveryDialog->show();
m_chatroomDiscoveryDialog->raise();
}
void
Controller::onSettingsAction()
{
m_settingDialog->setNick(QString(m_nick.c_str()));
m_settingDialog->show();
m_settingDialog->raise();
}
void
Controller::onProfileEditorAction()
{
m_profileEditor->show();
m_profileEditor->raise();
}
void
Controller::onAddContactAction()
{
emit refreshBrowseContact();
m_browseContactDialog->show();
m_browseContactDialog->raise();
}
void
Controller::onContactListAction()
{
m_contactPanel->show();
m_contactPanel->raise();
}
void
Controller::onDirectAdd()
{
m_addContactPanel->show();
m_addContactPanel->raise();
}
void
Controller::onUpdateLocalPrefixAction()
{
// Name interestName();
Interest interest("/localhop/ndn-autoconf/routable-prefixes");
interest.setInterestLifetime(time::milliseconds(1000));
interest.setMustBeFresh(true);
m_face->expressInterest(interest,
bind(&Controller::onLocalPrefix, this, _1, _2),
bind(&Controller::onLocalPrefixTimeout, this, _1));
}
void
Controller::onMinimizeAction()
{
m_settingDialog->hide();
m_startChatDialog->hide();
m_profileEditor->hide();
m_invitationDialog->hide();
m_addContactPanel->hide();
ChatDialogList::iterator it = m_chatDialogList.begin();
ChatDialogList::iterator end = m_chatDialogList.end();
for (; it != end; it++)
it->second->hide();
}
void
Controller::onQuitAction()
{
while (!m_chatDialogList.empty()) {
ChatDialogList::const_iterator it = m_chatDialogList.begin();
onRemoveChatDialog(QString::fromStdString(it->first));
}
if (m_invitationListenerId != 0)
m_face->unsetInterestFilter(m_invitationListenerId);
delete m_settingDialog;
delete m_startChatDialog;
delete m_profileEditor;
delete m_invitationDialog;
delete m_browseContactDialog;
delete m_addContactPanel;
m_face->getIoService().stop();
QApplication::quit();
}
void
Controller::onStartChatroom(const QString& chatroomName, bool secured)
{
Name chatroomPrefix;
chatroomPrefix.append("ndn")
.append("broadcast")
.append("ChronoChat")
.append(chatroomName.toStdString());
// check if the chatroom exists
if (m_chatDialogList.find(chatroomName.toStdString()) != m_chatDialogList.end()) {
QMessageBox::information(this, tr("ChronoChat"),
tr("You are creating an existing chatroom."
"You can check it in the context memu."));
return;
}
// TODO: We should create a chatroom specific key/cert
//(which should be created in the first half of this method
//, but let's use the default one for now.
// std::cout << "start chat room localprefix: " << m_localPrefix.toUri() << std::endl;
shared_ptr<IdentityCertificate> idCert
= m_keyChain.getCertificate(m_keyChain.getDefaultCertificateNameForIdentity(m_identity));
ChatDialog* chatDialog
= new ChatDialog(&m_contactManager, m_face, *idCert, chatroomPrefix
, m_localPrefix, m_nick, secured);
addChatDialog(chatroomName, chatDialog);
chatDialog->show();
}
void
Controller::onInvitationResponded(const ndn::Name& invitationName, bool accepted)
{
shared_ptr<Data> response = make_shared<Data>();
shared_ptr<IdentityCertificate> chatroomCert;
// generate reply;
if (accepted) {
Name responseName = invitationName;
responseName.append(m_localPrefix.wireEncode());
response->setName(responseName);
// We should create a particular certificate for this chatroom,
//but let's use default one for now.
chatroomCert
= m_keyChain.getCertificate(m_keyChain.getDefaultCertificateNameForIdentity(m_identity));
response->setContent(chatroomCert->wireEncode());
response->setFreshnessPeriod(time::milliseconds(1000));
}
else {
response->setName(invitationName);
response->setFreshnessPeriod(time::milliseconds(1000));
}
m_keyChain.signByIdentity(*response, m_identity);
// Check if we need a wrapper
Name invitationRoutingPrefix = getInvitationRoutingPrefix();
if (invitationRoutingPrefix.isPrefixOf(m_identity))
m_face->put(*response);
else {
Name wrappedName;
wrappedName.append(invitationRoutingPrefix)
.append(ROUTING_PREFIX_SEPARATOR, 2)
.append(response->getName());
// _LOG_DEBUG("onInvitationResponded: prepare reply " << wrappedName);
shared_ptr<Data> wrappedData = make_shared<Data>(wrappedName);
wrappedData->setContent(response->wireEncode());
wrappedData->setFreshnessPeriod(time::milliseconds(1000));
m_keyChain.signByIdentity(*wrappedData, m_identity);
m_face->put(*wrappedData);
}
// create chatroom
if (accepted) {
Invitation invitation(invitationName);
Name chatroomPrefix;
chatroomPrefix.append("ndn")
.append("broadcast")
.append("ChronoChat")
.append(invitation.getChatroom());
//We should create a chatroom specific key/cert
//(which should be created in the first half of this method,
//but let's use the default one for now.
shared_ptr<IdentityCertificate> idCert
= m_keyChain.getCertificate(m_keyChain.getDefaultCertificateNameForIdentity(m_identity));
ChatDialog* chatDialog
= new ChatDialog(&m_contactManager, m_face, *idCert,
chatroomPrefix, m_localPrefix, m_nick, true);
chatDialog->addSyncAnchor(invitation);
addChatDialog(QString::fromStdString(invitation.getChatroom()), chatDialog);
chatDialog->show();
}
}
void
Controller::onShowChatMessage(const QString& chatroomName, const QString& from, const QString& data)
{
m_trayIcon->showMessage(QString("Chatroom %1 has a new message").arg(chatroomName),
QString("<%1>: %2").arg(from).arg(data),
QSystemTrayIcon::Information, 20000);
m_trayIcon->setIcon(QIcon(":/images/note.png"));
}
void
Controller::onResetIcon()
{
m_trayIcon->setIcon(QIcon(":/images/icon_small.png"));
}
void
Controller::onRemoveChatDialog(const QString& chatroomName)
{
ChatDialogList::iterator it = m_chatDialogList.find(chatroomName.toStdString());
if (it != m_chatDialogList.end()) {
ChatDialog* deletedChat = it->second;
if (deletedChat)
delete deletedChat;
m_chatDialogList.erase(it);
m_discoveryLogic.removeLocalChatroom(Name::Component(chatroomName.toStdString()));
QAction* chatAction = m_chatActionList[chatroomName.toStdString()];
QAction* closeAction = m_closeActionList[chatroomName.toStdString()];
if (chatAction)
delete chatAction;
if (closeAction)
delete closeAction;
m_chatActionList.erase(chatroomName.toStdString());
m_closeActionList.erase(chatroomName.toStdString());
updateMenu();
}
}
void
Controller::onWarning(const QString& msg)
{
QMessageBox::information(this, tr("ChronoChat"), msg);
}
void
Controller::onError(const QString& msg)
{
QMessageBox::critical(this, tr("ChronoChat"), msg, QMessageBox::Ok);
exit(1);
}
void
Controller::onInvitationInterest(const ndn::Name& prefix,
const ndn::Interest& interest,
size_t routingPrefixOffset)
{
// _LOG_DEBUG("onInvitationInterest: " << interest.getName());
shared_ptr<Interest> invitationInterest =
make_shared<Interest>(boost::cref(interest.getName().getSubName(routingPrefixOffset)));
// check if the chatroom already exists;
try {
Invitation invitation(invitationInterest->getName());
if (m_chatDialogList.find(invitation.getChatroom()) != m_chatDialogList.end())
return;
}
catch (Invitation::Error& e) {
// Cannot parse the invitation;
return;
}
OnInterestValidated onValidated = bind(&Controller::onInvitationValidated, this, _1);
OnInterestValidationFailed onValidationFailed = bind(&Controller::onInvitationValidationFailed,
this, _1, _2);
m_validator.validate(*invitationInterest, onValidated, onValidationFailed);
}
void
Controller::onRosterChanged(const chronos::ChatroomInfo& info)
{
m_discoveryLogic.addLocalChatroom(info);
}
} // namespace chronos
#if WAF
#include "controller.moc"
// #include "controller.cpp.moc"
#endif