twisterp2pnetworkbittorrentblockchainipv6microbloggingdecentralizedsocial-networkdhttwister-ipv6twister-coretwisterarmyp2p-networktwister-server
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
232 lines
6.5 KiB
232 lines
6.5 KiB
#include "twister_utils.h" |
|
|
|
#include <libtorrent/session.hpp> |
|
#include <libtorrent/bencode.hpp> |
|
|
|
#include <boost/foreach.hpp> |
|
|
|
#include <stdio.h> |
|
|
|
using namespace std; |
|
using namespace boost; |
|
|
|
twister_utils::twister_utils() |
|
{ |
|
} |
|
|
|
int load_file(std::string const& filename, std::vector<char>& v, int limit) |
|
{ |
|
FILE* f = fopen(filename.c_str(), "rb"); |
|
if (f == NULL) |
|
return -1; |
|
int r = fseek(f, 0, SEEK_END); |
|
if (r != 0) { |
|
fclose(f); |
|
return -1; |
|
} |
|
long s = ftell(f); |
|
if (s < 0) { |
|
fclose(f); |
|
return -1; |
|
} |
|
|
|
if (s > limit) { |
|
fclose(f); |
|
return -2; |
|
} |
|
|
|
r = fseek(f, 0, SEEK_SET); |
|
if (r != 0) { |
|
fclose(f); |
|
return -1; |
|
} |
|
|
|
v.resize(s); |
|
if (s == 0) { |
|
fclose(f); |
|
return 0; |
|
} |
|
|
|
r = fread(&v[0], 1, v.size(), f); |
|
if (r < 0) { |
|
fclose(f); |
|
return -1; |
|
} |
|
|
|
fclose(f); |
|
|
|
if (r != s) return -3; |
|
|
|
return 0; |
|
} |
|
|
|
int save_file(std::string const& filename, std::vector<char>& v) |
|
{ |
|
using namespace libtorrent; |
|
|
|
// TODO: don't use internal file type here. use fopen() |
|
file f; |
|
error_code ec; |
|
if (!f.open(filename, file::write_only, ec)) return -1; |
|
if (ec) return -1; |
|
file::iovec_t b = {&v[0], v.size()}; |
|
size_type written = f.writev(0, &b, 1, ec); |
|
if (written != int(v.size())) return -3; |
|
if (ec) return -3; |
|
return 0; |
|
} |
|
|
|
using namespace json_spirit; |
|
using namespace libtorrent; |
|
|
|
Value entryToJson(const entry &e) |
|
{ |
|
Array lst; |
|
Object o; |
|
switch( e.type() ) { |
|
case entry::int_t: |
|
return e.integer(); |
|
case entry::string_t: |
|
return e.string(); |
|
case entry::list_t: |
|
for (entry::list_type::const_iterator i = e.list().begin(); i != e.list().end(); ++i) { |
|
lst.push_back( entryToJson(*i) ); |
|
} |
|
return lst; |
|
case entry::dictionary_t: |
|
for (entry::dictionary_type::const_iterator i = e.dict().begin(); i != e.dict().end(); ++i) { |
|
o.push_back(Pair(i->first, entryToJson(i->second))); |
|
} |
|
return o; |
|
default: |
|
return string("<uninitialized>"); |
|
} |
|
} |
|
|
|
entry jsonToEntry(const Value &v) |
|
{ |
|
entry::list_type lst; |
|
entry::dictionary_type dict; |
|
|
|
switch( v.type() ) { |
|
case int_type: |
|
return v.get_int(); |
|
case str_type: |
|
return v.get_str(); |
|
case array_type: |
|
for (Array::const_iterator i = v.get_array().begin(); i != v.get_array().end(); ++i) { |
|
lst.push_back( jsonToEntry(*i) ); |
|
} |
|
return lst; |
|
case obj_type: |
|
for (Object::const_iterator i = v.get_obj().begin(); i != v.get_obj().end(); ++i) { |
|
dict[ i->name_ ] = jsonToEntry(i->value_); |
|
} |
|
return dict; |
|
default: |
|
return string("<uninitialized>"); |
|
} |
|
} |
|
|
|
|
|
int saveUserData(std::string const& filename, std::map<std::string,UserData> const &users) |
|
{ |
|
entry userDict; |
|
|
|
std::map<std::string,UserData>::const_iterator i; |
|
for (i = users.begin(); i != users.end(); ++i) { |
|
UserData const &udata = i->second; |
|
|
|
if( udata.m_following.size() ) { |
|
entry &userData = userDict[i->first]; |
|
entry &followingList = userData["following"]; |
|
BOOST_FOREACH( std::string const &n, udata.m_following) { |
|
followingList.list().push_back(n); |
|
} |
|
} |
|
|
|
if( udata.m_directmsg.size() ) { |
|
entry &userData = userDict[i->first]; |
|
entry &dmDict = userData["dm"]; |
|
|
|
std::map<std::string, std::vector<StoredDirectMsg> >::const_iterator j; |
|
for (j = udata.m_directmsg.begin(); j != udata.m_directmsg.end(); ++j) { |
|
entry &dmList = dmDict[j->first]; |
|
BOOST_FOREACH( StoredDirectMsg const &stoDm, j->second) { |
|
entry dmElem; |
|
dmElem["time"] = stoDm.m_utcTime; |
|
dmElem["text"] = stoDm.m_text; |
|
dmElem["fromMe"] = stoDm.m_fromMe; |
|
dmList.list().push_back(dmElem); |
|
} |
|
} |
|
} |
|
} |
|
|
|
std::vector<char> buf; |
|
if( userDict.type() == entry::dictionary_t ) { |
|
bencode(std::back_inserter(buf), userDict); |
|
return save_file(filename, buf); |
|
} else { |
|
return 0; |
|
} |
|
} |
|
|
|
|
|
int loadUserData(std::string const& filename, std::map<std::string,UserData> &users) |
|
{ |
|
std::vector<char> in; |
|
if (load_file(filename, in) == 0) { |
|
lazy_entry userDict; |
|
error_code ec; |
|
if (lazy_bdecode(&in[0], &in[0] + in.size(), userDict, ec) == 0) { |
|
if( userDict.type() != lazy_entry::dict_t ) goto data_error; |
|
|
|
for( int i = 0; i < userDict.dict_size(); i++) { |
|
UserData udata; |
|
|
|
const lazy_entry *userData = userDict.dict_at(i).second; |
|
if( userData->type() != lazy_entry::dict_t ) goto data_error; |
|
|
|
const lazy_entry *followingList = userData->dict_find("following"); |
|
if( followingList ) { |
|
if( followingList->type() != lazy_entry::list_t ) goto data_error; |
|
|
|
for( int j = 0; j < followingList->list_size(); j++ ) { |
|
udata.m_following.insert( followingList->list_string_value_at(j) ); |
|
} |
|
} |
|
|
|
const lazy_entry *dmDict = userData->dict_find("dm"); |
|
if( dmDict ) { |
|
if( dmDict->type() != lazy_entry::dict_t ) goto data_error; |
|
|
|
for( int j = 0; j < dmDict->dict_size(); j++ ) { |
|
const lazy_entry *dmList = dmDict->dict_at(j).second; |
|
if( !dmList || dmList->type() != lazy_entry::list_t ) goto data_error; |
|
|
|
for( int k = 0; k < dmList->list_size(); k++ ) { |
|
const lazy_entry *dmElem = dmList->list_at(k); |
|
if( dmElem->type() != lazy_entry::dict_t ) goto data_error; |
|
|
|
StoredDirectMsg stoDm; |
|
stoDm.m_text = dmElem->dict_find_string_value("text"); |
|
stoDm.m_utcTime = dmElem->dict_find_int_value("time"); |
|
stoDm.m_fromMe = dmElem->dict_find_int_value("fromMe"); |
|
udata.m_directmsg[dmDict->dict_at(j).first].push_back(stoDm); |
|
} |
|
} |
|
} |
|
users[userDict.dict_at(i).first] = udata; |
|
} |
|
return 0; |
|
} |
|
} |
|
return -1; |
|
|
|
data_error: |
|
printf("loadUserData: unexpected bencode type - user_data corrupt!\n"); |
|
return -2; |
|
} |
|
|
|
|
|
|