#include "MSEHandshake.h" #include #include #include "Exception.h" #include "util.h" #include "prefs.h" #include "Socket.h" #include "Option.h" #include "DownloadContext.h" #include "FileEntry.h" #include "array_fun.h" #include "bittorrent_helper.h" namespace aria2 { class MSEHandshakeTest:public CppUnit::TestFixture { CPPUNIT_TEST_SUITE(MSEHandshakeTest); CPPUNIT_TEST(testHandshake); CPPUNIT_TEST_SUITE_END(); private: SharedHandle dctx_; void doHandshake(const SharedHandle& initiator, const SharedHandle& receiver); public: void setUp() { dctx_.reset(new DownloadContext()); unsigned char infoHash[20]; memset(infoHash, 0, sizeof(infoHash)); SharedHandle torrentAttrs(new TorrentAttribute()); torrentAttrs->infoHash = std::string(vbegin(infoHash), vend(infoHash)); dctx_->setAttribute(bittorrent::BITTORRENT, torrentAttrs); } void testHandshake(); }; CPPUNIT_TEST_SUITE_REGISTRATION(MSEHandshakeTest); namespace { std::pair, SharedHandle > createSocketPair() { SharedHandle initiatorSock(new SocketCore()); SocketCore receiverServerSock; receiverServerSock.bind(0); receiverServerSock.beginListen(); std::pair receiverAddrInfo; receiverServerSock.getAddrInfo(receiverAddrInfo); initiatorSock->establishConnection("localhost", receiverAddrInfo.second); initiatorSock->setBlockingMode(); SharedHandle receiverSock(receiverServerSock.acceptConnection()); receiverSock->setBlockingMode(); return std::pair, SharedHandle >(initiatorSock, receiverSock); } } // namespace void MSEHandshakeTest::doHandshake(const SharedHandle& initiator, const SharedHandle& receiver) { initiator->sendPublicKey(); while(initiator->getWantWrite()) { initiator->send(); } while(!receiver->receivePublicKey()) { receiver->read(); } receiver->sendPublicKey(); while(receiver->getWantWrite()) { receiver->send(); } while(!initiator->receivePublicKey()) { initiator->read(); } initiator->initCipher(bittorrent::getInfoHash(dctx_)); initiator->sendInitiatorStep2(); while(initiator->getWantWrite()) { initiator->send(); } while(!receiver->findReceiverHashMarker()) { receiver->read(); } std::vector > contexts; contexts.push_back(dctx_); while(!receiver->receiveReceiverHashAndPadCLength(contexts)) { receiver->read(); } while(!receiver->receivePad()) { receiver->read(); } while(!receiver->receiveReceiverIALength()) { receiver->read(); } while(!receiver->receiveReceiverIA()) { receiver->read(); } receiver->sendReceiverStep2(); while(receiver->getWantWrite()) { receiver->send(); } while(!initiator->findInitiatorVCMarker()) { initiator->read(); } while(!initiator->receiveInitiatorCryptoSelectAndPadDLength()) { initiator->read(); } while(!initiator->receivePad()) { initiator->read(); } } namespace { SharedHandle createMSEHandshake(SharedHandle socket, bool initiator, const Option* option) { SharedHandle h(new MSEHandshake(1, socket, option)); h->initEncryptionFacility(initiator); return h; } } // namespace void MSEHandshakeTest::testHandshake() { { Option op; op.put(PREF_BT_MIN_CRYPTO_LEVEL, V_PLAIN); std::pair, SharedHandle > sockPair = createSocketPair(); SharedHandle initiator = createMSEHandshake(sockPair.first, true, &op); SharedHandle receiver = createMSEHandshake(sockPair.second, false, &op); doHandshake(initiator, receiver); CPPUNIT_ASSERT_EQUAL(MSEHandshake::CRYPTO_PLAIN_TEXT, initiator->getNegotiatedCryptoType()); CPPUNIT_ASSERT_EQUAL(MSEHandshake::CRYPTO_PLAIN_TEXT, receiver->getNegotiatedCryptoType()); } { Option op; op.put(PREF_BT_MIN_CRYPTO_LEVEL, V_ARC4); std::pair, SharedHandle > sockPair = createSocketPair(); SharedHandle initiator = createMSEHandshake(sockPair.first, true, &op); SharedHandle receiver = createMSEHandshake(sockPair.second, false, &op); doHandshake(initiator, receiver); CPPUNIT_ASSERT_EQUAL(MSEHandshake::CRYPTO_ARC4, initiator->getNegotiatedCryptoType()); CPPUNIT_ASSERT_EQUAL(MSEHandshake::CRYPTO_ARC4, receiver->getNegotiatedCryptoType()); } } } // namespace aria2