/* -*- 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
