blob: ecec1258d5849c2e0873c52e00d509d82a3de02e [file] [log] [blame]
/* -*- 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 ChronoShare, a decentralized file sharing application over NDN.
*
* ChronoShare 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.
*
* ChronoShare 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 copies of the GNU General Public License along with
* ChronoShare, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*
* See AUTHORS.md for complete list of ChronoShare authors and contributors.
*/
#include "state-server.hpp"
#include "core/logging.hpp"
#include <ndn-cxx/util/digest.hpp>
#include <ndn-cxx/util/string-helper.hpp>
#include <boost/asio/io_service.hpp>
#include <boost/date_time/posix_time/posix_time.hpp>
#include <boost/filesystem/fstream.hpp>
#include <boost/lexical_cast.hpp>
namespace ndn {
namespace chronoshare {
_LOG_INIT(StateServer);
namespace fs = boost::filesystem;
StateServer::StateServer(Face& face, ActionLogPtr actionLog, const fs::path& rootDir,
const Name& userName, const std::string& sharedFolderName,
const name::Component& appName, ObjectManager& objectManager,
KeyChain& keyChain, time::milliseconds freshness)
: m_face(face)
, m_actionLog(actionLog)
, m_objectManager(objectManager)
, m_rootDir(rootDir)
, m_freshness(freshness)
, m_userName(userName)
, m_sharedFolderName(sharedFolderName)
, m_appName(appName)
, m_keyChain(keyChain)
, m_ioService(m_face.getIoService())
{
// may be later /localhop should be replaced with /%C1.M.S.localhost
// <PREFIX_INFO> = /localhop/<user's-device-name>/"chronoshare"/"info"
m_PREFIX_INFO = Name("/localhop");
m_PREFIX_INFO.append(m_userName).append(m_appName).append(m_sharedFolderName).append("info");
// <PREFIX_CMD> = /localhop/<user's-device-name>/"chronoshare"/"cmd"
m_PREFIX_CMD = Name("/localhop");
m_PREFIX_CMD.append(m_userName).append(m_appName).append(m_sharedFolderName).append("cmd");
registerPrefixes();
}
StateServer::~StateServer()
{
deregisterPrefixes();
}
void
StateServer::registerPrefixes()
{
// currently supporting limited number of command.
// will be extended to support all planned commands later
// <PREFIX_INFO>/"actions"/"all"/<segment> get list of all actions
Name actionsFolder = Name(m_PREFIX_INFO);
actionsFolder.append("actions").append("folder");
actionsFolderId =
m_face.setInterestFilter(InterestFilter(actionsFolder),
bind(&StateServer::info_actions_folder, this, _1, _2),
RegisterPrefixSuccessCallback(), RegisterPrefixFailureCallback());
_LOG_DEBUG("Register Prefix: " << actionsFolder);
Name actionsFile = Name(m_PREFIX_INFO);
actionsFile.append("actions").append("file");
actionsFileId =
m_face.setInterestFilter(InterestFilter(actionsFile),
bind(&StateServer::info_actions_file, this, _1, _2),
RegisterPrefixSuccessCallback(), RegisterPrefixFailureCallback());
_LOG_DEBUG("Register Prefix: " << actionsFile);
// <PREFIX_INFO>/"filestate"/"all"/<segment>
Name filesFolder = Name(m_PREFIX_INFO);
filesFolder.append("files").append("folder");
filesFolderId =
m_face.setInterestFilter(InterestFilter(filesFolder),
bind(&StateServer::info_files_folder, this, _1, _2),
RegisterPrefixSuccessCallback(), RegisterPrefixFailureCallback());
_LOG_DEBUG("Register Prefix: " << filesFolder);
// <PREFIX_CMD>/"restore"/"file"/<one-component-relative-file-name>/<version>/<file-hash>
Name restoreFile = Name(m_PREFIX_CMD);
restoreFile.append("restore").append("file");
restoreFileId =
m_face.setInterestFilter(InterestFilter(restoreFile),
bind(&StateServer::cmd_restore_file, this, _1, _2),
RegisterPrefixSuccessCallback(), RegisterPrefixFailureCallback());
_LOG_DEBUG("Register Prefix: " << restoreFile);
}
void
StateServer::deregisterPrefixes()
{
m_face.unsetInterestFilter(actionsFolderId);
m_face.unsetInterestFilter(actionsFileId);
m_face.unsetInterestFilter(filesFolderId);
m_face.unsetInterestFilter(restoreFileId);
}
void
StateServer::formatActionJson(json_spirit::Array& actions, const Name& name, sqlite3_int64 seq_no,
const ActionItem& action)
{
/*
* {
* "id": {
* "userName": "<NDN-NAME-OF-THE-USER>",
* "seqNo": "<SEQ_NO_OF_THE_ACTION>"
* },
* "timestamp": "<ACTION-TIMESTAMP>",
* "filename": "<FILENAME>",
*
* "action": "UPDATE | DELETE",
*
* // only if update
* "update": {
* "hash": "<FILE-HASH>",
* "timestamp": "<FILE-TIMESTAMP>",
* "chmod": "<FILE-MODE>",
* "segNum": "<NUMBER-OF-SEGMENTS(~file size)>"
* },
*
* // if parent_device_name is set
* "parentId": {
* "userName": "<NDN-NAME-OF-THE-USER>",
* "seqNo": "<SEQ_NO_OF_THE_ACTION>"
* }
* }
*/
using namespace json_spirit;
using namespace boost::posix_time;
Object json;
Object id;
id.push_back(Pair("userName", boost::lexical_cast<std::string>(name)));
id.push_back(Pair("seqNo", static_cast<int64_t>(seq_no)));
json.push_back(Pair("id", id));
json.push_back(Pair("timestamp", to_iso_extended_string(from_time_t(action.timestamp()))));
json.push_back(Pair("filename", action.filename()));
json.push_back(Pair("version", action.version()));
json.push_back(Pair("action", (action.action() == 0) ? "UPDATE" : "DELETE"));
if (action.action() == 0) {
Object update;
const Buffer hash(action.file_hash().c_str(), action.file_hash().size());
update.push_back(Pair("hash", toHex(hash)));
update.push_back(Pair("timestamp", to_iso_extended_string(from_time_t(action.mtime()))));
std::ostringstream chmod;
chmod << std::setbase(8) << std::setfill('0') << std::setw(4) << action.mode();
update.push_back(Pair("chmod", chmod.str()));
update.push_back(Pair("segNum", action.seg_num()));
json.push_back(Pair("update", update));
}
if (action.has_parent_device_name()) {
Object parentId;
Name parent_device_name(action.parent_device_name());
id.push_back(Pair("userName", boost::lexical_cast<std::string>(parent_device_name)));
id.push_back(Pair("seqNo", action.parent_seq_no()));
json.push_back(Pair("parentId", parentId));
}
actions.push_back(json);
}
void
StateServer::info_actions_folder(const InterestFilter& interesFilter, const Interest& interestTrue)
{
Name interest = interestTrue.getName();
_LOG_DEBUG(">> info_actions_folder: " << interest);
if (interest.size() - m_PREFIX_INFO.size() != 3 && interest.size() - m_PREFIX_INFO.size() != 4) {
_LOG_DEBUG("Invalid interest: " << interest);
return;
}
m_ioService.post(bind(&StateServer::info_actions_fileOrFolder_Execute, this, interest, true));
}
void
StateServer::info_actions_file(const InterestFilter& interesFilter, const Interest& interestTrue)
{
Name interest = interestTrue.getName();
if (interest.size() - m_PREFIX_INFO.size() != 3 && interest.size() - m_PREFIX_INFO.size() != 4) {
_LOG_DEBUG("Invalid interest: " << interest);
return;
}
_LOG_DEBUG(">> info_actions_file: " << interest);
m_ioService.post(bind(&StateServer::info_actions_fileOrFolder_Execute, this, interest, false));
}
void
StateServer::info_actions_fileOrFolder_Execute(const Name& interest, bool isFolder /* = true*/)
{
// <PREFIX_INFO>/"actions"/"folder|file"/<folder|file>/<offset> get list of all actions
if (interest.size() < 1) {
// ignore any unexpected interests and errors
_LOG_ERROR("empty interest name");
return;
}
uint64_t offset = interest.get(-1).toNumber();
/// @todo !!! add security checking
std::string fileOrFolderName;
if (interest.size() - m_PREFIX_INFO.size() == 4)
fileOrFolderName = interest.get(-2).toUri();
else // == 3
fileOrFolderName = "";
/*
* {
* "actions": [
* ...
* ],
*
* // only if there are more actions available
* "more": "<NDN-NAME-OF-NEXT-SEGMENT-OF-ACTION>"
* }
*/
_LOG_DEBUG("info_actions_fileOrFolder_Execute! offset: " << offset);
using namespace json_spirit;
Object json;
Array actions;
bool more;
if (isFolder) {
more = m_actionLog->LookupActionsInFolderRecursively(bind(StateServer::formatActionJson,
boost::ref(actions), _1, _2, _3),
fileOrFolderName, offset * 10, 10);
}
else {
more = m_actionLog->LookupActionsForFile(bind(StateServer::formatActionJson,
boost::ref(actions), _1, _2, _3),
fileOrFolderName, offset * 10, 10);
}
json.push_back(Pair("actions", actions));
if (more) {
json.push_back(Pair("more", boost::lexical_cast<std::string>(offset + 1)));
// Name more = Name(interest.getPartialName(0, interest.size() - 1))(offset + 1);
// json.push_back(Pair("more", boost::lexical_cast<std::string>(more)));
}
std::ostringstream os;
write_stream(Value(json), os, pretty_print | raw_utf8);
shared_ptr<Data> data = make_shared<Data>();
data->setName(interest);
data->setFreshnessPeriod(m_freshness);
data->setContent(reinterpret_cast<const uint8_t*>(os.str().c_str()), os.str().size());
m_keyChain.sign(*data);
m_face.put(*data);
}
void
StateServer::formatFilestateJson(json_spirit::Array& files, const FileItem& file)
{
/**
* {
* "filestate": [
* {
* "filename": "<FILENAME>",
* "owner": {
* "userName": "<NDN-NAME-OF-THE-USER>",
* "seqNo": "<SEQ_NO_OF_THE_ACTION>"
* },
*
* "hash": "<FILE-HASH>",
* "timestamp": "<FILE-TIMESTAMP>",
* "chmod": "<FILE-MODE>",
* "segNum": "<NUMBER-OF-SEGMENTS(~file size)>"
* }, ...,
* ]
*
* // only if there are more actions available
* "more": "<NDN-NAME-OF-NEXT-SEGMENT-OF-FILESTATE>"
* }
*/
using namespace json_spirit;
using namespace boost::posix_time;
Object json;
json.push_back(Pair("filename", file.filename()));
json.push_back(Pair("version", file.version()));
{
Object owner;
Name device_name(file.device_name());
owner.push_back(Pair("userName", boost::lexical_cast<std::string>(device_name)));
owner.push_back(Pair("seqNo", file.seq_no()));
json.push_back(Pair("owner", owner));
}
json.push_back(Pair("hash", toHex(reinterpret_cast<const uint8_t*>(file.file_hash().data()),
file.file_hash().size())));
json.push_back(Pair("timestamp", to_iso_extended_string(from_time_t(file.mtime()))));
std::ostringstream chmod;
chmod << std::setbase(8) << std::setfill('0') << std::setw(4) << file.mode();
json.push_back(Pair("chmod", chmod.str()));
json.push_back(Pair("segNum", file.seg_num()));
files.push_back(json);
}
void
debugFileState(const FileItem& file)
{
std::cout << file.filename() << std::endl;
}
void
StateServer::info_files_folder(const InterestFilter& interesFilter, const Interest& interestTrue)
{
Name interest = interestTrue.getName();
if (interest.size() - m_PREFIX_INFO.size() != 3 && interest.size() - m_PREFIX_INFO.size() != 4) {
_LOG_DEBUG("Invalid interest: " << interest << ", " << interest.size() - m_PREFIX_INFO.size());
return;
}
_LOG_DEBUG(">> info_files_folder: " << interest);
m_ioService.post(bind(&StateServer::info_files_folder_Execute, this, interest));
}
void
StateServer::info_files_folder_Execute(const Name& interest)
{
// <PREFIX_INFO>/"filestate"/"folder"/<one-component-relative-folder-name>/<offset>
if (interest.size() < 1) {
// ignore any unexpected interests and errors
_LOG_ERROR("empty interest name");
return;
}
uint64_t offset = interest.get(-1).toNumber();
// /// @todo !!! add security checking
std::string folder;
if (interest.size() - m_PREFIX_INFO.size() == 4)
folder = interest.get(-2).toUri();
else // == 3
folder = "";
/*
*{
* "files": [
* ...
* ],
*
* // only if there are more actions available
* "more": "<NDN-NAME-OF-NEXT-SEGMENT-OF-ACTION>"
*}
*/
using namespace json_spirit;
Object json;
Array files;
bool more =
m_actionLog->GetFileState()->LookupFilesInFolderRecursively(bind(StateServer::formatFilestateJson,
boost::ref(files), _1),
folder, offset * 10, 10);
json.push_back(Pair("files", files));
if (more) {
json.push_back(Pair("more", boost::lexical_cast<std::string>(offset + 1)));
// Name more = Name(interest.getPartialName(0, interest.size() - 1))(offset + 1);
// json.push_back(Pair("more", boost::lexical_cast<std::string>(more)));
}
std::ostringstream os;
write_stream(Value(json), os, pretty_print | raw_utf8);
shared_ptr<Data> data = make_shared<Data>();
data->setName(interest);
data->setFreshnessPeriod(m_freshness);
data->setContent(reinterpret_cast<const uint8_t*>(os.str().c_str()), os.str().size());
m_keyChain.sign(*data);
m_face.put(*data);
}
void
StateServer::cmd_restore_file(const InterestFilter& interesFilter, const Interest& interestTrue)
{
Name interest = interestTrue.getName();
if (interest.size() - m_PREFIX_CMD.size() != 4 && interest.size() - m_PREFIX_CMD.size() != 5) {
_LOG_DEBUG("Invalid interest: " << interest);
return;
}
_LOG_DEBUG(">> cmd_restore_file: " << interest);
m_ioService.post(bind(&StateServer::cmd_restore_file_Execute, this, interest));
}
void
StateServer::cmd_restore_file_Execute(const Name& interest)
{
// <PREFIX_CMD>/"restore"/"file"/<one-component-relative-file-name>/<version>/<file-hash>
/// @todo !!! add security checking
FileItemPtr file;
if (interest.size() - m_PREFIX_CMD.size() == 5) {
const Buffer hash(interest.get(-1).value(), interest.get(-1).value_size());
uint64_t version = interest.get(-2).toNumber();
std::string filename = interest.get(-3).toUri(); // should be safe even with full relative path
_LOG_DEBUG("filename: " << filename << " version: " << version);
file = m_actionLog->LookupAction(filename, version, hash);
if (!file) {
_LOG_ERROR("Requested file is not found: [" << filename << "] version [" << version << "] hash ["
<< toHex(hash)
<< "]");
}
}
else {
uint64_t version = interest.get(-1).toNumber();
std::string filename = interest.get(-2).toUri();
file = m_actionLog->LookupAction(filename, version, Buffer(0, 0));
if (!file) {
_LOG_ERROR("Requested file is not found: [" << filename << "] version [" << version << "]");
}
}
if (!file) {
shared_ptr<Data> data = make_shared<Data>();
data->setName(interest);
data->setFreshnessPeriod(m_freshness);
std::string msg = "FAIL: Requested file is not found";
data->setContent(reinterpret_cast<const uint8_t*>(msg.c_str()), msg.size());
m_keyChain.sign(*data);
m_face.put(*data);
return;
}
const Buffer hash(file->file_hash().c_str(), file->file_hash().size());
///////////////////
// now the magic //
///////////////////
fs::path filePath = m_rootDir / file->filename();
Name deviceName(
Block((const unsigned char*)file->device_name().c_str(), file->device_name().size()));
_LOG_DEBUG("filePath" << filePath << " deviceName " << deviceName);
try {
if (fs::exists(filePath) && fs::last_write_time(filePath) == file->mtime()
#if BOOST_VERSION >= 104900
&&
fs::status(filePath).permissions() == static_cast<fs::perms>(file->mode())
#endif
) {
fs::ifstream input(filePath, std::ios::in | std::ios::binary);
if (*util::Sha256(input).computeDigest() == hash) {
shared_ptr<Data> data = make_shared<Data>();
data->setName(interest);
data->setFreshnessPeriod(m_freshness);
std::string msg = "OK: File already exists";
data->setContent(reinterpret_cast<const uint8_t*>(msg.c_str()), msg.size());
m_keyChain.sign(*data);
m_face.put(*data);
_LOG_DEBUG("Asking to assemble a file, but file already exists on a filesystem");
return;
}
}
}
catch (fs::filesystem_error& error) {
shared_ptr<Data> data = make_shared<Data>();
data->setName(interest);
data->setFreshnessPeriod(m_freshness);
std::string msg = "FAIL: File operation failed";
data->setContent(reinterpret_cast<const uint8_t*>(msg.c_str()), msg.size());
m_keyChain.sign(*data);
m_face.put(*data);
_LOG_ERROR("File operations failed on [" << filePath << "](ignoring)");
}
_LOG_TRACE("Restoring file [" << filePath << "]"
<< " deviceName "
<< deviceName);
if (m_objectManager.objectsToLocalFile(deviceName, hash, filePath)) {
last_write_time(filePath, file->mtime());
#if BOOST_VERSION >= 104900
permissions(filePath, static_cast<fs::perms>(file->mode()));
#endif
shared_ptr<Data> data = make_shared<Data>();
data->setName(interest);
data->setFreshnessPeriod(m_freshness);
std::string msg = "OK";
data->setContent(reinterpret_cast<const uint8_t*>(msg.c_str()), msg.size());
m_keyChain.sign(*data);
m_face.put(*data);
_LOG_DEBUG("Restoring file successfully!");
}
else {
shared_ptr<Data> data = make_shared<Data>();
data->setName(interest);
data->setFreshnessPeriod(m_freshness);
std::string msg = "FAIL: Unknown error while restoring file";
data->setContent(reinterpret_cast<const uint8_t*>(msg.c_str()), msg.size());
m_keyChain.sign(*data);
m_face.put(*data);
}
}
} // namespace chronoshare
} // namespace ndn