aboutsummaryrefslogtreecommitdiff
path: root/src/main.cpp
diff options
context:
space:
mode:
authordec05eba <dec05eba@protonmail.com>2018-05-16 10:24:22 +0200
committerdec05eba <dec05eba@protonmail.com>2018-05-16 10:25:06 +0200
commite6331c04af99d7deeb9b15be02dd30665c3c41ce (patch)
tree7d8c1dadcdefe4579fbcc750a352f3b80879b347 /src/main.cpp
parent777d8053540cf94f5b2156c8e2b242efb4d07007 (diff)
Rewrite code to work with new backend
Redesigned join channel system, read README.md for more information
Diffstat (limited to 'src/main.cpp')
-rw-r--r--src/main.cpp397
1 files changed, 232 insertions, 165 deletions
diff --git a/src/main.cpp b/src/main.cpp
index 18f10fe..249c5d4 100644
--- a/src/main.cpp
+++ b/src/main.cpp
@@ -18,6 +18,7 @@
#include <odhtdb/bin2hex.hpp>
#include <odhtdb/hex2bin.hpp>
#include <ntp/NtpClient.hpp>
+#include <sibs/SafeSerializer.hpp>
#include <X11/Xlib.h>
using namespace std;
@@ -128,19 +129,6 @@ void channelAddStoredMessage(Channel *channel, const odhtdb::Hash &requestHash,
}
}
-void channelAddStoredMessages(Channel *channel, const odhtdb::DatabaseStorageObjectList *nodeStorage)
-{
- printf("Load %u message(s) in channel %s\n", nodeStorage->objects.size(), channel->getName().c_str());
- for(auto nodeStorageAddedObject : nodeStorage->objects)
- {
- if(nodeStorageAddedObject->decryptedObject.operation == odhtdb::DatabaseOperation::ADD_DATA)
- {
- StringView decryptedObject((const char*)nodeStorageAddedObject->decryptedObject.data.data, nodeStorageAddedObject->decryptedObject.data.size);
- channelAddStoredMessage(channel, nodeStorageAddedObject->requestHash, nodeStorageAddedObject->creatorPublicKey, decryptedObject, nodeStorageAddedObject->createdTimestamp);
- }
- }
-}
-
int main(int argc, char **argv)
{
/*
@@ -158,8 +146,6 @@ int main(int argc, char **argv)
window.setVerticalSyncEnabled(false);
window.setFramerateLimit(FRAMERATE_FOCUSED);
- odhtdb::Database database("bootstrap.ring.cx", 4222, Cache::getDchatDir());
-
//Video video(500, 500, "https://www.youtube.com/watch?v=bs0-EX9mJmg");
Cache cache;
@@ -171,51 +157,69 @@ int main(int argc, char **argv)
vector<Channel*> channels;
- odhtdb::Signature::KeyPair *currentUserKeyPair = nullptr;
- vector<odhtdb::NodeLocalUser> localNodeUsers;
vector<odhtdb::DatabaseNode> waitingToJoinChannels;
- string currentUserName;
- string currentUserPassword;
+ odhtdb::MapHash<odhtdb::StoredNodeInfo> localNodeUsers;
+ string currentUsername;
+ string currentPassword;
recursive_mutex channelMessageMutex;
bool waitingToJoin = false;
+ bool loggedIn = false;
+
+ odhtdb::Database *database = nullptr;
+ odhtdb::DatabaseCallbackFuncs callbackFuncs;
- database.setOnCreateNodeCallback([&waitingToJoinChannels, &database, &channels, &channelMessageMutex, &waitingToJoin](const odhtdb::DatabaseCreateNodeRequest &request)
+ callbackFuncs.createNodeCallbackFunc = [&waitingToJoinChannels, &database, &channels, &channelMessageMutex, &waitingToJoin, &localNodeUsers](const odhtdb::DatabaseCreateNodeRequest &request)
{
lock_guard<recursive_mutex> lock(channelMessageMutex);
+
+ auto nodeUserData = localNodeUsers.find(*request.nodeHash);
+ if(nodeUserData == localNodeUsers.end()) return;
+
+ User *localUser;
+ if(nodeUserData->second.userKeyPair->getPublicKey() == *request.creatorPublicKey)
+ localUser = new OnlineLocalUser("NoName", *nodeUserData->second.userKeyPair);
+ else
+ localUser = new OfflineUser("You");
+
+ shared_ptr<odhtdb::Hash> databaseNodeHash = make_shared<odhtdb::Hash>();
+ memcpy(databaseNodeHash->getData(), request.nodeHash->getData(), odhtdb::HASH_BYTE_SIZE);
+ odhtdb::DatabaseNode databaseNode(nodeUserData->second.nodeEncryptionKey, databaseNodeHash);
+ Channel *channel = new Channel("NoChannelName", databaseNode, localUser, database);
+ ChannelSidePanel::addChannel(channel);
+ channels.push_back(channel);
+ Channel::setCurrent(channel);
+
+ if(localUser->type == User::Type::OFFLINE)
+ {
+ User *nodeCreatorUser = new OnlineRemoteUser("NoName", *request.creatorPublicKey);
+ channel->addUserLocally(nodeCreatorUser);
+ }
+
for(vector<odhtdb::DatabaseNode>::iterator it = waitingToJoinChannels.begin(); it != waitingToJoinChannels.end(); ++it)
{
if(*request.nodeHash == *it->getRequestHash())
{
- User *localUser = new OfflineUser("You");
- Channel *channel = new Channel(request.name, *it, localUser, &database);
- ChannelSidePanel::addChannel(channel);
- channels.push_back(channel);
- Channel::setCurrent(channel);
-
- User *nodeCreatorUser = new OnlineUser(request.creatorUser);
- channel->addUserLocally(nodeCreatorUser);
waitingToJoin = true;
-
waitingToJoinChannels.erase(it);
return;
}
}
- });
+ };
- database.setOnAddNodeCallback([&channels, &channelMessageMutex](const odhtdb::DatabaseAddNodeRequest &request)
+ callbackFuncs.addNodeCallbackFunc = [&channels, &channelMessageMutex](const odhtdb::DatabaseAddNodeRequest &request)
{
lock_guard<recursive_mutex> lock(channelMessageMutex);
for(Channel *channel : channels)
{
if(*request.nodeHash == *channel->getNodeInfo().getRequestHash())
{
- channelAddStoredMessage(channel, *request.requestHash, request.creatorUser->getPublicKey(), StringView((const char*)request.decryptedData.data, request.decryptedData.size), request.timestamp);
+ channelAddStoredMessage(channel, *request.requestHash, *request.creatorPublicKey, StringView((const char*)request.decryptedData.data, request.decryptedData.size), request.timestamp);
return;
}
}
- });
+ };
- database.setOnAddUserCallback([&currentUserKeyPair, &channels, &channelMessageMutex, &waitingToJoin, &database](const odhtdb::DatabaseAddUserRequest &request)
+ callbackFuncs.addUserCallbackFunc = [&channels, &channelMessageMutex, &waitingToJoin, &localNodeUsers](const odhtdb::DatabaseAddUserRequest &request)
{
lock_guard<recursive_mutex> lock(channelMessageMutex);
printf("Add user callback. Channel to add user to: %s\n", request.nodeHash->toString().c_str());
@@ -225,34 +229,35 @@ int main(int argc, char **argv)
if(*request.nodeHash == *channel->getNodeInfo().getRequestHash())
{
printf("Add user to one of my channels\n");
- if(currentUserKeyPair && request.userToAdd->getPublicKey() == currentUserKeyPair->getPublicKey() && channel->getLocalUser()->type != User::Type::ONLINE)
+ auto nodeUserData = localNodeUsers.find(*request.nodeHash);
+
+ User *userToAdd = channel->getUserByPublicKey(*request.userToAddPublicKey);
+ if(userToAdd)
{
- // Replace remote user with local user, if we are the remote user
- database.getStorage().getLocalNodeUsers(*currentUserKeyPair);
- auto userToAdd = database.getStorage().getUserByPublicKey(*request.nodeHash, currentUserKeyPair->getPublicKey());
- printf("You were added to channel %s by %s\n", request.nodeHash->toString().c_str(), request.creatorUser->getName().c_str());
- channel->replaceLocalUser(new OnlineUser(userToAdd));
- waitingToJoin = false;
+ fprintf(stderr, "User %s already exists in channel\n", request.userToAddPublicKey->toString().c_str());
return;
}
- User *userToAdd = channel->getUserByPublicKey(request.userToAdd->getPublicKey());
- if(userToAdd)
+ if(*request.userToAddPublicKey == nodeUserData->second.userKeyPair->getPublicKey())
{
- fprintf(stderr, "User %s already exists in channel\n", request.userToAdd->getPublicKey().toString().c_str());
+ printf("You were added to channel %s by %s\n", request.nodeHash->toString().c_str(), request.creatorPublicKey->toString().c_str());
+ channel->replaceLocalUser(new OnlineLocalUser("NoName", *nodeUserData->second.userKeyPair));
+ waitingToJoin = false;
return;
}
- User *newRemoteUser = new OnlineUser(request.userToAdd);
+ User *newRemoteUser = new OnlineRemoteUser("NoName", *request.userToAddPublicKey);
channel->addUserLocally(newRemoteUser);
return;
}
}
- });
+ };
+ database = new odhtdb::Database("bootstrap.ring.cx", 4222, Cache::getDchatDir(), callbackFuncs);
// Login to account
- Command::add("login", [&currentUserKeyPair, &currentUserName, &currentUserPassword, &localNodeUsers, &database, &channels, &channelMessageMutex](const vector<string> &args)
+ Command::add("login", [&localNodeUsers, &database, &channels, &loggedIn, &channelMessageMutex, &currentUsername, &currentPassword](const vector<string> &args)
{
+ lock_guard<recursive_mutex> lock(channelMessageMutex);
if(args.size() != 2)
{
fprintf(stderr, "Expected 2 arguments for command login (username and password), got %u argument(s)\n", args.size());
@@ -261,8 +266,7 @@ int main(int argc, char **argv)
try
{
- odhtdb::Signature::KeyPair keyPair = database.getStorage().decryptLocalEncryptedUser(args[0], args[1]);
- localNodeUsers = database.getStorage().getLocalNodeUsers(keyPair);
+ localNodeUsers = database->getStoredNodeUserInfoDecrypted(args[0], args[1]);
ChannelSidePanel::removeAllChannels();
for(Channel *channel : channels)
@@ -271,51 +275,25 @@ int main(int argc, char **argv)
}
channels.clear();
- lock_guard<recursive_mutex> lock(channelMessageMutex);
- for(auto localNodeUser : localNodeUsers)
+ printf("Loading %u channel(s) for user\n", localNodeUsers.size());
+ for(auto &localNodeUser : localNodeUsers)
{
- auto nodeStorage = database.getStorage().getStorage(localNodeUser.nodeHash);
- if(!nodeStorage) continue;
-
- auto nodeDecryptionKeyResult = database.getStorage().getNodeDecryptionKey(localNodeUser.nodeHash);
- if(!nodeDecryptionKeyResult.first) continue;
-
- User *newLocalUser = new OnlineUser(localNodeUser.localUser);
- odhtdb::DatabaseNode databaseNode(nodeDecryptionKeyResult.second, make_shared<odhtdb::Hash>(localNodeUser.nodeHash));
- Channel *channel = new Channel(nodeStorage->nodeName, databaseNode, newLocalUser, &database);
-
- auto nodeUserMapByPublicKey = database.getStorage().getNodeUsers(localNodeUser.nodeHash);
- for(auto nodeUserIt : *nodeUserMapByPublicKey)
- {
- if(nodeUserIt.second != localNodeUser.localUser)
- {
- User *newRemoteUser = new OnlineUser(nodeUserIt.second);
- channel->addUserLocally(newRemoteUser);
- }
- }
-
- ChannelSidePanel::addChannel(channel);
- channels.push_back(channel);
- Channel::setCurrent(channel);
- channelAddStoredMessages(channel, nodeStorage);
+ database->loadNode(localNodeUser.first);
}
printf("Successfully logged into user %s\n", args[0].c_str());
- if(currentUserKeyPair)
- delete currentUserKeyPair;
- currentUserKeyPair = new odhtdb::Signature::KeyPair(keyPair);
-
- currentUserName = args[0];
- currentUserPassword = args[1];
+ currentUsername = args[0];
+ currentPassword = args[1];
+ loggedIn = true;
}
- catch(odhtdb::DatabaseStorageException &e)
+ catch(std::exception &e)
{
fprintf(stderr, "Failed to login, reason: %s\n", e.what());
}
});
// Register account
- Command::add("register", [&currentUserKeyPair, &currentUserName, &currentUserPassword, &localNodeUsers, &database, &channelMessageMutex](const vector<string> &args)
+ Command::add("register", [&localNodeUsers, &database, &channels, &loggedIn, &channelMessageMutex, &currentUsername, &currentPassword](const vector<string> &args)
{
lock_guard<recursive_mutex> lock(channelMessageMutex);
if(args.size() != 2)
@@ -324,33 +302,34 @@ int main(int argc, char **argv)
return;
}
- if(currentUserKeyPair)
+ try
{
- fprintf(stderr, "You can't register a new account when you are logged in, please logout first\n");
- return;
+ database->storeUserWithoutNodes(args[0], args[1]);
}
-
- odhtdb::Signature::KeyPair keyPair;
- if(!database.getStorage().storeLocalUser(args[0], keyPair, args[1]))
+ catch(odhtdb::SqlExecException &e)
{
fprintf(stderr, "User with name %s already exists in storage\n", args[0].c_str());
return;
}
- printf("Registered user %s, public key: %s, private key: %s\n", args[0].c_str(), keyPair.getPublicKey().toString().c_str(), keyPair.getPrivateKey().toString().c_str());
- printf("Successfully logged into user %s\n", args[0].c_str());
+ printf("Registered user %s\n", args[0].c_str());
- if(currentUserKeyPair)
- delete currentUserKeyPair;
- currentUserKeyPair = new odhtdb::Signature::KeyPair(keyPair);
+ ChannelSidePanel::removeAllChannels();
+ for(Channel *channel : channels)
+ {
+ delete channel;
+ }
+ channels.clear();
localNodeUsers.clear();
- currentUserName = args[0];
- currentUserPassword = args[1];
+ printf("Successfully logged into user %s\n", args[0].c_str());
+ currentUsername = args[0];
+ currentPassword = args[1];
+ loggedIn = true;
});
// Create channel
- Command::add("cc", [&currentUserKeyPair, &currentUserName, &database, &channels, &channelMessageMutex](const vector<string> &args)
+ Command::add("cc", [&database, &channels, &channelMessageMutex, &loggedIn, &localNodeUsers, &currentUsername, &currentPassword](const vector<string> &args)
{
lock_guard<recursive_mutex> lock(channelMessageMutex);
if(args.size() != 1)
@@ -359,69 +338,113 @@ int main(int argc, char **argv)
return;
}
- if(!currentUserKeyPair)
+ if(!loggedIn)
{
fprintf(stderr, "You are not logged in. Please login before creating a channel\n");
return;
}
- auto createResponse = database.create(currentUserName, *currentUserKeyPair, args[0]);
- database.commit();
+ auto createResponse = database->create();
printf("Created database '%s', join key: '%s'\n", args[0].c_str(), createChannelJoinKey(createResponse).c_str());
- User *newLocalUser = new OnlineUser(createResponse->getNodeAdminUser());
+ User *newLocalUser = new OnlineLocalUser("NoName", *createResponse->getNodeAdminKeyPair());
odhtdb::DatabaseNode databaseNode(createResponse->getNodeEncryptionKey(), createResponse->getRequestHash());
- Channel *channel = new Channel(args[0], databaseNode, newLocalUser, &database);
+ Channel *channel = new Channel(args[0], databaseNode, newLocalUser, database);
ChannelSidePanel::addChannel(channel);
channels.push_back(channel);
Channel::setCurrent(channel);
+
+ localNodeUsers[*createResponse->getRequestHash()] = { createResponse->getNodeEncryptionKey(), createResponse->getNodeAdminKeyPair() };
+ database->storeNodeInfoForUserEncrypted(databaseNode, currentUsername, currentPassword, *createResponse->getNodeAdminKeyPair());
+ Channel::getCurrent()->addLocalMessage("Channel created and stored in database", Channel::getCurrent()->getSystemUser());
});
- // Add user
- Command::add("au", [&offlineChannel, &channelMessageMutex](const vector<string> &args)
+ // Create invite key
+ Command::add("invite", [&channelMessageMutex, &offlineChannel, &database](const vector<string> &args)
{
lock_guard<recursive_mutex> lock(channelMessageMutex);
- if(args.size() != 2)
- {
- fprintf(stderr, "Expected 2 arguments for command au (user id, group id), got %u argument(s)\n", args.size());
- return;
- }
-
- if(args[0].size() != odhtdb::PUBLIC_KEY_NUM_BYTES * 2)
- {
- fprintf(stderr, "User id is wrong size. Expected to be %u characters, was %u character(s)\n", odhtdb::PUBLIC_KEY_NUM_BYTES * 2, args[0].size());
- return;
- }
-
- if(args[1].size() != odhtdb::GROUP_ID_LENGTH * 2)
+ if(args.size() != 0)
{
- fprintf(stderr, "Group id is wrong size. Expected to be %u characters, was %u character(s)\n", odhtdb::GROUP_ID_LENGTH * 2, args[1].size());
+ fprintf(stderr, "Expected 0 arguments for command invite, got %u argument(s)\n", args.size());
return;
}
Channel *currentChannel = Channel::getCurrent();
if(currentChannel == &offlineChannel)
{
- Channel::getCurrent()->addLocalMessage("You need to be in a channel to add user to the channel", Channel::getCurrent()->getSystemUser());
+ Channel::getCurrent()->addLocalMessage("You need to be in a channel to create an invite key", Channel::getCurrent()->getSystemUser());
return;
}
- auto userIdRaw = odhtdb::hex2bin(args[0].c_str(), args[0].size());
- odhtdb::Signature::PublicKey userPublicKey(userIdRaw.data(), userIdRaw.size());
+ // TODO: Verify user has permission to add users before generating invite key, otherwise invite will fail and the users attempting to join will wait in futile
- auto groupIdRaw = odhtdb::hex2bin(args[1].c_str(), args[1].size());
- bool userAddResult = currentChannel->addUser(userPublicKey, groupIdRaw);
- if(userAddResult)
- {
- printf("Added user to your channel!\n");
- }
- else
+ auto channelNodeHash = currentChannel->getNodeInfo().getRequestHash();
+ auto channelEncryptionKey = currentChannel->getNodeInfo().getNodeEncryptionKey();
+ shared_ptr<odhtdb::OwnedMemory> encryptionKey = make_shared<odhtdb::OwnedMemory>(new u8[odhtdb::ENCRYPTION_KEY_BYTE_SIZE], odhtdb::ENCRYPTION_KEY_BYTE_SIZE);
+ odhtdb::Encryption::generateKey((unsigned char*)encryptionKey->data);
+
+ string inviteKey = odhtdb::bin2hex((const char*)channelNodeHash->getData(), channelNodeHash->getSize());
+ inviteKey += '&';
+ inviteKey += odhtdb::bin2hex((const char*)encryptionKey->data, odhtdb::ENCRYPTION_KEY_BYTE_SIZE);
+
+ string msg = "You are now listening for users to join the channel using the key: ";
+ msg += inviteKey;
+
+ Channel::getCurrent()->addLocalMessage(msg, Channel::getCurrent()->getSystemUser());
+ printf("%s\n", msg.c_str());
+
+ sibs::SafeSerializer keySerializer;
+ keySerializer.add((const u8*)channelNodeHash->getData(), channelNodeHash->getSize());
+ keySerializer.add((const u8*)encryptionKey->data, odhtdb::ENCRYPTION_KEY_BYTE_SIZE);
+ dht::InfoHash key = odhtdb::Database::getInfoHash(keySerializer.getBuffer().data(), keySerializer.getBuffer().size());
+ database->receiveCustomMessage(key, [&channelMessageMutex, encryptionKey, channelEncryptionKey, currentChannel, &database](const void *data, usize size)
{
- fprintf(stderr, "Failed to add user to your channel!\n");
- }
+ // TODO: User can remove channel @currentChannel before we get here, meaning @currentChannel is deleted and would be invalid; causing the program to crash
+ try
+ {
+ sibs::SafeDeserializer deserializer((const u8*)data, size);
+ u8 userToAddPublicKeyRaw[odhtdb::PUBLIC_KEY_NUM_BYTES];
+ deserializer.extract(userToAddPublicKeyRaw, odhtdb::PUBLIC_KEY_NUM_BYTES);
+ odhtdb::Signature::PublicKey userToAddPublicKey((const char*)userToAddPublicKeyRaw, odhtdb::PUBLIC_KEY_NUM_BYTES);
+ string unsignedEncryptedMsg = userToAddPublicKey.unsign(odhtdb::DataView((void*)deserializer.getBuffer(), deserializer.getSize()));
+
+ sibs::SafeDeserializer encryptedDataDeserializer((const u8*)unsignedEncryptedMsg.data(), unsignedEncryptedMsg.size());
+ u8 nonce[odhtdb::ENCRYPTION_NONCE_BYTE_SIZE];
+ encryptedDataDeserializer.extract(nonce, odhtdb::ENCRYPTION_NONCE_BYTE_SIZE);
+ odhtdb::Decryption decryptedMsg(odhtdb::DataView((void*)encryptedDataDeserializer.getBuffer(), encryptedDataDeserializer.getSize()),
+ odhtdb::DataView(nonce, odhtdb::ENCRYPTION_NONCE_BYTE_SIZE),
+ encryptionKey->getView());
+
+ // TODO: Create GUI for accepting users into channel instead of accepting ALL users
+ sibs::SafeSerializer encryptedDataSerializer;
+ encryptedDataSerializer.add((const u8*)channelEncryptionKey->data, channelEncryptionKey->size);
+ encryptedDataSerializer.add((const u8*)userToAddPublicKey.getData(), userToAddPublicKey.getSize());
+ odhtdb::Encryption encryptedChannelKey(odhtdb::DataView((void*)encryptedDataSerializer.getBuffer().data(), encryptedDataSerializer.getBuffer().size()), encryptionKey->getView());
+
+ sibs::SafeSerializer serializer;
+ serializer.add((const u8*)encryptedChannelKey.getNonce().data, encryptedChannelKey.getNonce().size);
+ serializer.add((const u8*)encryptedChannelKey.getCipherText().data, encryptedChannelKey.getCipherText().size);
+ const auto &localUserPublicKey = static_cast<OnlineLocalUser*>(currentChannel->getLocalUser())->getPublicKey();
+ auto localUserGroups = database->getUserGroups(*currentChannel->getNodeInfo().getRequestHash(), localUserPublicKey);
+ if(!localUserGroups.empty())
+ {
+ fprintf(stderr, "No group to add user to...\n");
+ return sibs::SafeSerializer();
+ }
+ lock_guard<recursive_mutex> lock(channelMessageMutex);
+ currentChannel->addUser(localUserPublicKey, localUserGroups[0].getView());
+ return serializer;
+ }
+ catch(std::exception &e)
+ {
+ fprintf(stderr, "Failed while parsing user data to add to channel, reason: %s\n", e.what());
+ }
+ return sibs::SafeSerializer();
+ });
});
- Command::add("jc", [&currentUserKeyPair, &database, &localNodeUsers, &channelMessageMutex, &waitingToJoin, &waitingToJoinChannels](const vector<string> &args)
+ // Join channel using invite key
+ Command::add("jc", [&loggedIn, &database, &localNodeUsers, &channelMessageMutex, &waitingToJoin, &waitingToJoinChannels, &currentUsername, &currentPassword](const vector<string> &args)
{
lock_guard<recursive_mutex> lock(channelMessageMutex);
if(args.size() != 1)
@@ -436,29 +459,90 @@ int main(int argc, char **argv)
return;
}
- if(!currentUserKeyPair)
+ if(!loggedIn)
{
fprintf(stderr, "You are not logged in. Please login before joining a channel\n");
return;
}
- odhtdb::DatabaseNode databaseNode = createDatabaseNodeFromJoinKey(args[0]);
- for(auto localNodeUser : localNodeUsers)
+ string nodeHashBinRaw = odhtdb::hex2bin(args[0].c_str(), 64);
+ shared_ptr<odhtdb::Hash> nodeHash = make_shared<odhtdb::Hash>();
+ memcpy(nodeHash->getData(), nodeHashBinRaw.data(), nodeHashBinRaw.size());
+ auto nodeUserIt = localNodeUsers.find(*nodeHash);
+ if(nodeUserIt != localNodeUsers.end())
{
- if(*databaseNode.getRequestHash() == localNodeUser.nodeHash)
- {
- fprintf(stderr, "You have already joined the channel %s\n", databaseNode.getRequestHash()->toString().c_str());
- return;
- }
+ fprintf(stderr, "You have already joined the channel %s\n", args[0].c_str());
+ return;
}
- database.seed(databaseNode);
- waitingToJoinChannels.push_back(databaseNode);
+ string encryptionKeyBinRaw = odhtdb::hex2bin(args[0].c_str() + 65, 64);
+ shared_ptr<odhtdb::OwnedMemory> encryptionKey = make_shared<odhtdb::OwnedMemory>(new u8[odhtdb::ENCRYPTION_KEY_BYTE_SIZE], odhtdb::ENCRYPTION_KEY_BYTE_SIZE);
+ memcpy(encryptionKey->data, encryptionKeyBinRaw.data(), encryptionKeyBinRaw.size());
+
+ shared_ptr<odhtdb::Signature::KeyPair> keyPair = make_shared<odhtdb::Signature::KeyPair>();
+ sibs::SafeSerializer serializer;
+ serializer.add((const u8*)keyPair->getPublicKey().getData(), keyPair->getPublicKey().getSize());
+
+ localNodeUsers[*nodeHash] = { encryptionKey, keyPair };
+
+ const char *msg = "please let me join";
+ odhtdb::Encryption encryptedJoinMsg(odhtdb::DataView((void*)msg, strlen(msg)), encryptionKey->getView());
+ sibs::SafeSerializer encryptedDataSerializer;
+ encryptedDataSerializer.add((const u8*)encryptedJoinMsg.getNonce().data, encryptedJoinMsg.getNonce().size);
+ encryptedDataSerializer.add((const u8*)encryptedJoinMsg.getCipherText().data, encryptedJoinMsg.getCipherText().size);
+ string signedEncryptedMsg = keyPair->getPrivateKey().sign(odhtdb::DataView(encryptedDataSerializer.getBuffer().data(), encryptedDataSerializer.getBuffer().size()));
+ serializer.add((const u8*)signedEncryptedMsg.data(), signedEncryptedMsg.size());
+
+ sibs::SafeSerializer keySerializer;
+ keySerializer.add((const u8*)nodeHash->getData(), nodeHash->getSize());
+ keySerializer.add((const u8*)encryptionKeyBinRaw.data(), odhtdb::ENCRYPTION_KEY_BYTE_SIZE);
+ dht::InfoHash key = odhtdb::Database::getInfoHash(keySerializer.getBuffer().data(), keySerializer.getBuffer().size());
+ database->sendCustomMessage(key, move(serializer.getBuffer()), [&database, nodeHash, encryptionKey, &waitingToJoinChannels, &channelMessageMutex, keyPair, &currentUsername, &currentPassword, &localNodeUsers](bool gotResponse, const void *data, usize size)
+ {
+ if(!gotResponse)
+ {
+ printf("We didn't get a response from anybody in the channel. Is there nobody that can add us (nobody with the permission required to do so or no online users) or is the node hash invalid?\n");
+ return false;
+ }
+
+ try
+ {
+ sibs::SafeDeserializer deserializer((const u8*)data, size);
+ u8 nonce[odhtdb::ENCRYPTION_NONCE_BYTE_SIZE];
+ deserializer.extract(nonce, odhtdb::ENCRYPTION_NONCE_BYTE_SIZE);
+ odhtdb::Decryption decryptedMsg(odhtdb::DataView((void*)deserializer.getBuffer(), deserializer.getSize()),
+ odhtdb::DataView(nonce, odhtdb::ENCRYPTION_NONCE_BYTE_SIZE),
+ encryptionKey->getView());
+ if(decryptedMsg.getDecryptedText().size != odhtdb::ENCRYPTION_KEY_BYTE_SIZE + odhtdb::PUBLIC_KEY_NUM_BYTES)
+ {
+ fprintf(stderr, "Invite response was of unexpected size, maybe it wasn't mean for us?\n");
+ return true;
+ }
+
+ odhtdb::DataView channelEncryptionKey(decryptedMsg.getDecryptedText().data, odhtdb::ENCRYPTION_KEY_BYTE_SIZE);
+ odhtdb::DataView invitedUserPublicKey((u8*)decryptedMsg.getDecryptedText().data + odhtdb::ENCRYPTION_KEY_BYTE_SIZE, odhtdb::PUBLIC_KEY_NUM_BYTES);
+ if(memcmp(keyPair->getPublicKey().getData(), invitedUserPublicKey.data, odhtdb::PUBLIC_KEY_NUM_BYTES) != 0)
+ {
+ fprintf(stderr, "Invite response was not meant for us\n");
+ return true;
+ }
+
+ odhtdb::DatabaseNode databaseNode(encryptionKey, nodeHash);
+ database->storeNodeInfoForUserEncrypted(databaseNode, currentUsername, currentPassword, *keyPair);
+ printf("Got a response from a person in the channel, we might get added...\n");
+ waitingToJoinChannels.push_back(databaseNode);
+ lock_guard<recursive_mutex> lock(channelMessageMutex);
+ database->seed(databaseNode);
+ return false;
+ }
+ catch(std::exception &e)
+ {
+ fprintf(stderr, "Failed while parsing join response for invite link, reason: %s\n", e.what());
+ }
+ return true;
+
+ });
waitingToJoin = true;
- // TODO: Add the channel to join to a pending join list in a file and remove from it when we have joined the channel.
- // The reason for doing that is so if we crash or lose internet connection before we have got `create node` request from remote peers,
- // then we need to start seeding again when we login. Once we have `create node` request, then it's added to local cache and when you login,
- // it will be used to seed the channel.
});
// Scale UI
@@ -556,25 +640,8 @@ int main(int argc, char **argv)
Channel::getCurrent()->addLocalMessage(msg, Channel::getCurrent()->getSystemUser());
});
- // Get username and id (public key)
- Command::add("whoami", [&currentUserKeyPair, &currentUserName](const vector<string> &args)
- {
- if(!currentUserKeyPair)
- {
- Channel::getCurrent()->addLocalMessage("You are not logged in", Channel::getCurrent()->getSystemUser());
- return;
- }
-
- string response = "Username: ";
- response += currentUserName;
- response += ", id: ";
- response += currentUserKeyPair->getPublicKey().toString();
- printf("%s\n", response.c_str());
- Channel::getCurrent()->addLocalMessage(response, Channel::getCurrent()->getSystemUser());
- });
-
// Change nick of current user in current channel
- Command::add("nick", [&currentUserKeyPair, &offlineChannel](const vector<string> &args)
+ Command::add("nick", [&loggedIn, &offlineChannel](const vector<string> &args)
{
if(args.size() != 1)
{
@@ -585,7 +652,7 @@ int main(int argc, char **argv)
return;
}
- if(!currentUserKeyPair)
+ if(!loggedIn)
{
Channel::getCurrent()->addLocalMessage("You need to be logged in to change your nickname", Channel::getCurrent()->getSystemUser());
return;
@@ -610,7 +677,7 @@ int main(int argc, char **argv)
});
// Get channel join key
- Command::add("joinkey", [&currentUserKeyPair, &currentUserName, &offlineChannel](const vector<string> &args)
+ Command::add("joinkey", [&offlineChannel](const vector<string> &args)
{
Channel *currentChannel = Channel::getCurrent();
if(!currentChannel || currentChannel == &offlineChannel)