From 658ed6054acdce355d906f50e885c76c9b33b2fb Mon Sep 17 00:00:00 2001 From: Miguel Freitas Date: Tue, 5 Nov 2013 22:54:53 -0200 Subject: [PATCH 1/4] export number of known torrent peers (list_peers) to the dht tracker resource and use it as estimative to the number of followers. at least we get the number of *online* followers. --- .../include/libtorrent/kademlia/dht_get.hpp | 2 ++ .../libtorrent/kademlia/dht_tracker.hpp | 2 +- .../include/libtorrent/kademlia/node.hpp | 5 ++-- libtorrent/src/kademlia/dht_get.cpp | 28 ++++++++++++++++++- libtorrent/src/kademlia/dht_tracker.cpp | 4 +-- libtorrent/src/kademlia/node.cpp | 10 ++++--- libtorrent/src/torrent.cpp | 6 ++-- 7 files changed, 44 insertions(+), 13 deletions(-) diff --git a/libtorrent/include/libtorrent/kademlia/dht_get.hpp b/libtorrent/include/libtorrent/kademlia/dht_get.hpp index 6bac7357..0be2f554 100644 --- a/libtorrent/include/libtorrent/kademlia/dht_get.hpp +++ b/libtorrent/include/libtorrent/kademlia/dht_get.hpp @@ -98,6 +98,8 @@ private: bool m_done:1; bool m_got_data:1; bool m_justToken:1; + + friend class dht_get_observer; }; class dht_get_observer : public observer diff --git a/libtorrent/include/libtorrent/kademlia/dht_tracker.hpp b/libtorrent/include/libtorrent/kademlia/dht_tracker.hpp index ee8408ca..af580286 100644 --- a/libtorrent/include/libtorrent/kademlia/dht_tracker.hpp +++ b/libtorrent/include/libtorrent/kademlia/dht_tracker.hpp @@ -93,7 +93,7 @@ namespace libtorrent { namespace dht entry state() const; void announce(std::string const& trackerName, sha1_hash const& ih - , address addr, int listen_port, bool seed, bool myself + , address addr, int listen_port, bool seed, bool myself, int list_peers , boost::function const&)> f); void putData(std::string const &username, std::string const &resource, bool multi, diff --git a/libtorrent/include/libtorrent/kademlia/node.hpp b/libtorrent/include/libtorrent/kademlia/node.hpp index cb1fe9c4..231c8c3f 100644 --- a/libtorrent/include/libtorrent/kademlia/node.hpp +++ b/libtorrent/include/libtorrent/kademlia/node.hpp @@ -112,6 +112,7 @@ struct torrent_entry { std::string name; std::set peers; + int list_peers; // number of known peers (copied from torrent status) }; struct dht_storage_item @@ -215,7 +216,7 @@ public: #endif void announce(std::string const& trackerName, sha1_hash const& info_hash - , address addr, int listen_port, bool seed, bool myself + , address addr, int listen_port, bool seed, bool myself, int list_peers , boost::function const&)> f); void putData(std::string const &username, std::string const &resource, bool multi, @@ -268,7 +269,7 @@ protected: void lookup_peers(sha1_hash const& info_hash, int prefix, entry& reply , bool noseed, bool scrape) const; - void add_peer(std::string const& name, sha1_hash const& info_hash, address addr, int port, bool seed); + void add_peer(std::string const& name, sha1_hash const& info_hash, address addr, int port, bool seed, int list_peers); dht_settings const& m_settings; diff --git a/libtorrent/src/kademlia/dht_get.cpp b/libtorrent/src/kademlia/dht_get.cpp index 307fa6bd..d3b6224e 100644 --- a/libtorrent/src/kademlia/dht_get.cpp +++ b/libtorrent/src/kademlia/dht_get.cpp @@ -125,8 +125,34 @@ void dht_get_observer::reply(msg const& m) values_list.push_back(entry()); values_list.back() = *e; } - //printf("dht_get::reply from %s:%d with %d entries\n", m.addr.address().to_string().c_str(), m.addr.port(), values_list.size()); + //printf("dht_get::reply from %s:%d with %d entries\n", m.addr.address().to_string().c_str(), m.addr.port(), values_list.size()); static_cast(m_algorithm.get())->got_data(values_list); + } else { + // special case for trackers (non-signed content) + // pretend it is a normal dht resource to the caller + dht_get *dget( static_cast(m_algorithm.get()) ); + if( dget->m_targetResource == "tracker" && dget->m_multi ) { + int followers = r->dict_find_int_value("followers"); + if( followers ) { + entry::dictionary_type v; + v["followers"] = followers; + + entry::dictionary_type target; + target["n"] = dget->m_targetUser; + target["r"] = dget->m_targetResource; + target["t"] = dget->m_multi ? "m" : "s"; + + entry::dictionary_type p; + p["target"] = target; + p["v"] = v; + + entry::dictionary_type e; + e["p"] = p; + entry::list_type values_list; + values_list.push_back(e); + dget->got_data(values_list); + } + } } // look for nodes diff --git a/libtorrent/src/kademlia/dht_tracker.cpp b/libtorrent/src/kademlia/dht_tracker.cpp index 24459306..f6f451c0 100644 --- a/libtorrent/src/kademlia/dht_tracker.cpp +++ b/libtorrent/src/kademlia/dht_tracker.cpp @@ -427,10 +427,10 @@ namespace libtorrent { namespace dht } void dht_tracker::announce(std::string const& trackerName, sha1_hash const& ih - , address addr, int listen_port, bool seed, bool myself + , address addr, int listen_port, bool seed, bool myself, int list_peers , boost::function const&)> f) { - m_dht.announce(trackerName, ih, addr, listen_port, seed, myself, f); + m_dht.announce(trackerName, ih, addr, listen_port, seed, myself, list_peers, f); } void dht_tracker::putData(std::string const &username, std::string const &resource, bool multi, diff --git a/libtorrent/src/kademlia/node.cpp b/libtorrent/src/kademlia/node.cpp index f6fe01e7..26c07ae9 100644 --- a/libtorrent/src/kademlia/node.cpp +++ b/libtorrent/src/kademlia/node.cpp @@ -400,7 +400,7 @@ void node_impl::add_node(udp::endpoint node) m_rpc.invoke(e, node, o); } -void node_impl::announce(std::string const& trackerName, sha1_hash const& info_hash, address addr, int listen_port, bool seed, bool myself +void node_impl::announce(std::string const& trackerName, sha1_hash const& info_hash, address addr, int listen_port, bool seed, bool myself, int list_peers , boost::function const&)> f) { #ifdef TORRENT_DHT_VERBOSE_LOGGING @@ -410,7 +410,7 @@ void node_impl::announce(std::string const& trackerName, sha1_hash const& info_h // [MF] is_unspecified() is not always available. never mind. //if( !addr.is_unspecified() ) { - add_peer( trackerName, info_hash, addr, listen_port, seed ); + add_peer( trackerName, info_hash, addr, listen_port, seed, list_peers ); //} // do not announce other peers, just add them to our local m_map. @@ -762,6 +762,7 @@ void node_impl::lookup_peers(sha1_hash const& info_hash, int prefix, entry& repl torrent_entry const& v = i->second; if (!v.name.empty()) reply["n"] = v.name; + reply["followers"] = v.list_peers; if (scrape) { @@ -803,7 +804,7 @@ void node_impl::lookup_peers(sha1_hash const& info_hash, int prefix, entry& repl return; } -void node_impl::add_peer(std::string const &name, sha1_hash const& info_hash, address addr, int port, bool seed) +void node_impl::add_peer(std::string const &name, sha1_hash const& info_hash, address addr, int port, bool seed, int list_peers) { torrent_entry& v = m_map[info_hash]; @@ -814,6 +815,7 @@ void node_impl::add_peer(std::string const &name, sha1_hash const& info_hash, ad v.name = name; if (v.name.size() > 50) v.name.resize(50); } + if (list_peers) v.list_peers = list_peers; peer_entry peer; peer.addr = tcp::endpoint(addr, port); @@ -1144,7 +1146,7 @@ void node_impl::incoming_request(msg const& m, entry& e) } add_peer( msg_keys[3] ? msg_keys[3]->string_value() : std::string(), info_hash, - m.addr.address(), port, msg_keys[4] && msg_keys[4]->int_value()); + m.addr.address(), port, msg_keys[4] && msg_keys[4]->int_value(), 0); #ifdef TORRENT_DHT_VERBOSE_LOGGING ++g_announces; diff --git a/libtorrent/src/torrent.cpp b/libtorrent/src/torrent.cpp index 71597a6c..ac76de61 100644 --- a/libtorrent/src/torrent.cpp +++ b/libtorrent/src/torrent.cpp @@ -2073,7 +2073,7 @@ namespace libtorrent p->last_connected && (m_ses.session_time() - p->last_connected) < (4*3600); if( p->connectable && ( p->connection || connect_recently) ) { m_ses.m_dht->announce(name(), m_torrent_file->info_hash() - , p->address(), p->port, p->seed, false + , p->address(), p->port, p->seed, false, m_policy.num_peers() , boost::bind(&nop)); } } @@ -2086,8 +2086,8 @@ namespace libtorrent #endif boost::weak_ptr self(shared_from_this()); - m_ses.m_dht->announce(name(), m_torrent_file->info_hash() - , m_ses.external_address().external_address(address_v4()), port, is_seed(), true + m_ses.m_dht->announce(name(), m_torrent_file->info_hash() + , m_ses.external_address().external_address(address_v4()), port, is_seed(), true, m_policy.num_peers() , boost::bind(&torrent::on_dht_announce_response_disp, self, _1)); } From b7fb1f3d3cb1f0938a2f36cb75ba5786c8e300f4 Mon Sep 17 00:00:00 2001 From: Miguel Freitas Date: Tue, 5 Nov 2013 23:25:47 -0200 Subject: [PATCH 2/4] strange android/boost compilation fix --- libtorrent/src/kademlia/node.cpp | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/libtorrent/src/kademlia/node.cpp b/libtorrent/src/kademlia/node.cpp index 26c07ae9..0da27d71 100644 --- a/libtorrent/src/kademlia/node.cpp +++ b/libtorrent/src/kademlia/node.cpp @@ -556,13 +556,18 @@ bool node_impl::refresh_storage() { m_last_refreshed_item = m_storage_table.begin()->first; } + time_duration sleepToRefresh; if( num_refreshable ) { - m_next_storage_refresh = minutes(60) / num_refreshable + time_now(); + sleepToRefresh = minutes(60) / num_refreshable; } else { - m_next_storage_refresh = minutes(10) + time_now(); + sleepToRefresh = minutes(10); } + m_next_storage_refresh = time_now() + sleepToRefresh; - printf("node dht: next storage refresh in %d seconds\n", (m_next_storage_refresh - time_now())/1000000 ); +#ifndef __ANDROID__ + printf("node dht: next storage refresh in %d seconds\n", + static_cast(sleepToRefresh/1000000) ); +#endif return did_something; } From 8ec4b3e16ed1e125230e9ba4ae592a47765f042a Mon Sep 17 00:00:00 2001 From: Miguel Freitas Date: Wed, 6 Nov 2013 09:21:01 -0200 Subject: [PATCH 3/4] i can't get this conversion from time_duration to string right. i give up. --- libtorrent/src/kademlia/node.cpp | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/libtorrent/src/kademlia/node.cpp b/libtorrent/src/kademlia/node.cpp index 0da27d71..c3c9af6b 100644 --- a/libtorrent/src/kademlia/node.cpp +++ b/libtorrent/src/kademlia/node.cpp @@ -35,6 +35,7 @@ POSSIBILITY OF SUCH DAMAGE. #include #include #include +//#include #include "libtorrent/io.hpp" #include "libtorrent/bencode.hpp" @@ -564,10 +565,10 @@ bool node_impl::refresh_storage() { } m_next_storage_refresh = time_now() + sleepToRefresh; -#ifndef __ANDROID__ - printf("node dht: next storage refresh in %d seconds\n", - static_cast(sleepToRefresh/1000000) ); -#endif +/* + printf("node dht: next storage refresh in %s\n", + boost::posix_time::to_simple_string(sleepToRefresh).c_str() ); +*/ return did_something; } From 2c256b6edb534e378df7f80929c6a9e5b0c12fd2 Mon Sep 17 00:00:00 2001 From: Miguel Freitas Date: Wed, 6 Nov 2013 09:22:05 -0200 Subject: [PATCH 4/4] implement piece verification using old signatures. now i can reenable banning peers due to bad pieces. --- TODO | 14 ----------- libtorrent/src/torrent.cpp | 3 --- src/twister.cpp | 48 +++++++++++++++----------------------- src/twister.h | 4 ++-- 4 files changed, 21 insertions(+), 48 deletions(-) diff --git a/TODO b/TODO index fcea3716..95a9353a 100644 --- a/TODO +++ b/TODO @@ -1,17 +1,3 @@ -- Take care of posts using older public key when key is replaced. - -notes: not very difficult, GetTransaction must receive a maximum block number to search the -transaction (we get this from post["height"]). another txIndex should be set to speedup lookup -(key in db includes the number of the block that changed tx so previous one can be found). -pseudocode: - getTxIndex( key = "userX" ) => block h contains this tx; - while( h > max_h ) - getTxIndex( "userX_h" ) => block h contains the previous tx -=> GetTransation: new parameter maxHeight done! - -- Until old public key is properly used, disable banning torrent peers due to bad piece hashes. -note: torrent.cpp line 3286 (function piece_failed), iteration to ban peers is disabled (continue). - - Count UTF8 chars in acceptSignedPost to proper limit the 140 characters. - Encrypt user_data (which contains all DMs) diff --git a/libtorrent/src/torrent.cpp b/libtorrent/src/torrent.cpp index ac76de61..3b7f62a8 100644 --- a/libtorrent/src/torrent.cpp +++ b/libtorrent/src/torrent.cpp @@ -3308,9 +3308,6 @@ namespace libtorrent for (std::set::iterator i = peers.begin() , end(peers.end()); i != end; ++i) { - // [MF] FIXME FIXME: BANNING BY FAILED HASH DISABLED - READ TODO! - continue; - policy::peer* p = static_cast(*i); if (p == 0) continue; TORRENT_ASSERT(p->in_use); diff --git a/src/twister.cpp b/src/twister.cpp index 8784e6f5..f36515df 100644 --- a/src/twister.cpp +++ b/src/twister.cpp @@ -606,44 +606,33 @@ std::string createSignature(std::string const &strMessage, std::string const &st } -bool getUserPubKey(std::string const &strUsername, CPubKey &pubkey) +bool getUserPubKey(std::string const &strUsername, CPubKey &pubkey, int maxHeight) { - { - CKeyID keyID; - if( pwalletMain->GetKeyIdFromUsername(strUsername, keyID) ) { - if( !pwalletMain->GetPubKey(keyID, pubkey) ) { - // error? should not have failed. - } - } + CTransaction txOut; + uint256 hashBlock; + if( !GetTransaction(strUsername, txOut, hashBlock, maxHeight) ) { + //printf("getUserPubKey: user unknown '%s'\n", strUsername.c_str()); + return false; } + std::vector< std::vector > vData; + if( !txOut.pubKey.ExtractPushData(vData) || vData.size() < 1 ) { + printf("getUserPubKey: broken pubkey for user '%s'\n", strUsername.c_str()); + return false; + } + pubkey = CPubKey(vData[0]); if( !pubkey.IsValid() ) { - CTransaction txOut; - uint256 hashBlock; - if( !GetTransaction(strUsername, txOut, hashBlock) ) { - //printf("getUserPubKey: user unknown '%s'\n", strUsername.c_str()); - return false; - } - - std::vector< std::vector > vData; - if( !txOut.pubKey.ExtractPushData(vData) || vData.size() < 1 ) { - printf("getUserPubKey: broken pubkey for user '%s'\n", strUsername.c_str()); - return false; - } - pubkey = CPubKey(vData[0]); - if( !pubkey.IsValid() ) { - printf("getUserPubKey: invalid pubkey for user '%s'\n", strUsername.c_str()); - return false; - } + printf("getUserPubKey: invalid pubkey for user '%s'\n", strUsername.c_str()); + return false; } return true; } -bool verifySignature(std::string const &strMessage, std::string const &strUsername, std::string const &strSign) +bool verifySignature(std::string const &strMessage, std::string const &strUsername, std::string const &strSign, int maxHeight) { CPubKey pubkey; - if( !getUserPubKey(strUsername, pubkey) ) { + if( !getUserPubKey(strUsername, pubkey, maxHeight) ) { printf("verifySignature: no pubkey for user '%s'\n", strUsername.c_str()); return false; } @@ -761,7 +750,7 @@ bool acceptSignedPost(char const *data, int data_size, std::string username, int std::pair postbuf = post->data_section(); ret = verifySignature( std::string(postbuf.first,postbuf.second), - username, sig); + username, sig, height); if( !ret ) { sprintf(errbuf,"bad post signature"); } else { @@ -771,11 +760,12 @@ bool acceptSignedPost(char const *data, int data_size, std::string username, int if( rt ) { if( flags ) (*flags) |= USERPOST_FLAG_RT; std::string username_rt = rt->dict_find_string_value("n"); + int height_rt = rt->dict_find_int_value("height",-1); std::pair rtbuf = rt->data_section(); ret = verifySignature( std::string(rtbuf.first,rtbuf.second), - username_rt, sig_rt); + username_rt, sig_rt, height_rt); if( !ret ) { sprintf(errbuf,"bad RT signature"); } diff --git a/src/twister.h b/src/twister.h index 913b985d..a2b29e58 100644 --- a/src/twister.h +++ b/src/twister.h @@ -23,10 +23,10 @@ public: void startSessionTorrent(boost::thread_group& threadGroup); void stopSessionTorrent(); -bool getUserPubKey(std::string const &strUsername, CPubKey &pubkey); +bool getUserPubKey(std::string const &strUsername, CPubKey &pubkey, int maxHeight = -1); std::string createSignature(std::string const &strMessage, CKeyID &keyID); std::string createSignature(std::string const &strMessage, std::string const &strUsername); -bool verifySignature(std::string const &strMessage, std::string const &strUsername, std::string const &strSign); +bool verifySignature(std::string const &strMessage, std::string const &strUsername, std::string const &strSign, int maxHeight = -1); bool acceptSignedPost(char const *data, int data_size, std::string username, int seq, std::string &errmsg, boost::uint32_t *flags); bool validatePostNumberForUser(std::string const &username, int k);