UtilTest.cc 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913
  1. #include "util.h"
  2. #include <cstring>
  3. #include <string>
  4. #include <iostream>
  5. #include <cppunit/extensions/HelperMacros.h>
  6. #include "FixedNumberRandomizer.h"
  7. #include "DlAbortEx.h"
  8. #include "BitfieldMan.h"
  9. #include "ByteArrayDiskWriter.h"
  10. #include "FileEntry.h"
  11. #include "File.h"
  12. #include "array_fun.h"
  13. namespace aria2 {
  14. class UtilTest:public CppUnit::TestFixture {
  15. CPPUNIT_TEST_SUITE(UtilTest);
  16. CPPUNIT_TEST(testTrim);
  17. CPPUNIT_TEST(testSplit);
  18. CPPUNIT_TEST(testSplit_many);
  19. CPPUNIT_TEST(testEndsWith);
  20. CPPUNIT_TEST(testReplace);
  21. CPPUNIT_TEST(testStartsWith);
  22. // may be moved to other helper class in the future.
  23. CPPUNIT_TEST(testGetContentDispositionFilename);
  24. CPPUNIT_TEST(testRandomAlpha);
  25. CPPUNIT_TEST(testToUpper);
  26. CPPUNIT_TEST(testToLower);
  27. CPPUNIT_TEST(testUrldecode);
  28. CPPUNIT_TEST(testGetRealSize);
  29. CPPUNIT_TEST(testAbbrevSize);
  30. CPPUNIT_TEST(testToStream);
  31. CPPUNIT_TEST(testIsNumber);
  32. CPPUNIT_TEST(testIsLowercase);
  33. CPPUNIT_TEST(testIsUppercase);
  34. CPPUNIT_TEST(testAlphaToNum);
  35. CPPUNIT_TEST(testMkdirs);
  36. CPPUNIT_TEST(testConvertBitfield);
  37. CPPUNIT_TEST(testParseIntRange);
  38. CPPUNIT_TEST(testParseIntRange_invalidRange);
  39. CPPUNIT_TEST(testParseInt);
  40. CPPUNIT_TEST(testParseUInt);
  41. CPPUNIT_TEST(testParseLLInt);
  42. CPPUNIT_TEST(testParseULLInt);
  43. CPPUNIT_TEST(testToString_binaryStream);
  44. CPPUNIT_TEST(testItos);
  45. CPPUNIT_TEST(testUitos);
  46. CPPUNIT_TEST(testNtoh64);
  47. CPPUNIT_TEST(testUrlencode);
  48. CPPUNIT_TEST(testHtmlEscape);
  49. CPPUNIT_TEST(testJoinPath);
  50. CPPUNIT_TEST(testParseIndexPath);
  51. CPPUNIT_TEST(testCreateIndexPathMap);
  52. CPPUNIT_TEST(testGenerateRandomData);
  53. CPPUNIT_TEST(testFromHex);
  54. CPPUNIT_TEST(testParsePrioritizePieceRange);
  55. CPPUNIT_TEST(testApplyDir);
  56. CPPUNIT_TEST_SUITE_END();
  57. private:
  58. public:
  59. void setUp() {
  60. }
  61. void testTrim();
  62. void testSplit();
  63. void testSplit_many();
  64. void testEndsWith();
  65. void testReplace();
  66. void testStartsWith();
  67. // may be moved to other helper class in the future.
  68. void testGetContentDispositionFilename();
  69. void testRandomAlpha();
  70. void testToUpper();
  71. void testToLower();
  72. void testUrldecode();
  73. void testGetRealSize();
  74. void testAbbrevSize();
  75. void testToStream();
  76. void testIsNumber();
  77. void testIsLowercase();
  78. void testIsUppercase();
  79. void testAlphaToNum();
  80. void testMkdirs();
  81. void testConvertBitfield();
  82. void testParseIntRange();
  83. void testParseIntRange_invalidRange();
  84. void testParseInt();
  85. void testParseUInt();
  86. void testParseLLInt();
  87. void testParseULLInt();
  88. void testToString_binaryStream();
  89. void testItos();
  90. void testUitos();
  91. void testNtoh64();
  92. void testUrlencode();
  93. void testHtmlEscape();
  94. void testJoinPath();
  95. void testParseIndexPath();
  96. void testCreateIndexPathMap();
  97. void testGenerateRandomData();
  98. void testFromHex();
  99. void testParsePrioritizePieceRange();
  100. void testApplyDir();
  101. };
  102. CPPUNIT_TEST_SUITE_REGISTRATION( UtilTest );
  103. void UtilTest::testTrim() {
  104. std::string str1 = "aria2";
  105. CPPUNIT_ASSERT_EQUAL(str1, util::trim("aria2"));
  106. CPPUNIT_ASSERT_EQUAL(str1, util::trim(" aria2"));
  107. CPPUNIT_ASSERT_EQUAL(str1, util::trim(" aria2 "));
  108. CPPUNIT_ASSERT_EQUAL(str1, util::trim(" aria2 "));
  109. std::string str2 = "aria2 debut";
  110. CPPUNIT_ASSERT_EQUAL(str2, util::trim("aria2 debut"));
  111. CPPUNIT_ASSERT_EQUAL(str2, util::trim(" aria2 debut "));
  112. std::string str3 = "";
  113. CPPUNIT_ASSERT_EQUAL(str3, util::trim(""));
  114. CPPUNIT_ASSERT_EQUAL(str3, util::trim(" "));
  115. CPPUNIT_ASSERT_EQUAL(str3, util::trim(" "));
  116. std::string str4 = "A";
  117. CPPUNIT_ASSERT_EQUAL(str4, util::trim("A"));
  118. CPPUNIT_ASSERT_EQUAL(str4, util::trim(" A "));
  119. CPPUNIT_ASSERT_EQUAL(str4, util::trim(" A "));
  120. }
  121. void UtilTest::testSplit() {
  122. std::pair<std::string, std::string> p1;
  123. util::split(p1, "name=value", '=');
  124. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  125. CPPUNIT_ASSERT_EQUAL(std::string("value"), p1.second);
  126. util::split(p1, " name = value ", '=');
  127. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  128. CPPUNIT_ASSERT_EQUAL(std::string("value"), p1.second);
  129. util::split(p1, "=value", '=');
  130. CPPUNIT_ASSERT_EQUAL(std::string(""), p1.first);
  131. CPPUNIT_ASSERT_EQUAL(std::string("value"), p1.second);
  132. util::split(p1, "name=", '=');
  133. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  134. CPPUNIT_ASSERT_EQUAL(std::string(""), p1.second);
  135. util::split(p1, "name", '=');
  136. CPPUNIT_ASSERT_EQUAL(std::string("name"), p1.first);
  137. CPPUNIT_ASSERT_EQUAL(std::string(""), p1.second);
  138. }
  139. void UtilTest::testSplit_many() {
  140. std::deque<std::string> v1;
  141. util::split("name1=value1; name2=value2; name3=value3",std::back_inserter(v1),
  142. ";", true);
  143. CPPUNIT_ASSERT_EQUAL(3, (int)v1.size());
  144. std::deque<std::string>::iterator itr = v1.begin();
  145. CPPUNIT_ASSERT_EQUAL(std::string("name1=value1"), *itr++);
  146. CPPUNIT_ASSERT_EQUAL(std::string("name2=value2"), *itr++);
  147. CPPUNIT_ASSERT_EQUAL(std::string("name3=value3"), *itr++);
  148. v1.clear();
  149. util::split("name1=value1; name2=value2; name3=value3",std::back_inserter(v1),
  150. ";", false);
  151. CPPUNIT_ASSERT_EQUAL(3, (int)v1.size());
  152. itr = v1.begin();
  153. CPPUNIT_ASSERT_EQUAL(std::string("name1=value1"), *itr++);
  154. CPPUNIT_ASSERT_EQUAL(std::string(" name2=value2"), *itr++);
  155. CPPUNIT_ASSERT_EQUAL(std::string(" name3=value3"), *itr++);
  156. }
  157. void UtilTest::testEndsWith() {
  158. std::string target = "abcdefg";
  159. std::string part = "fg";
  160. CPPUNIT_ASSERT(util::endsWith(target, part));
  161. target = "abdefg";
  162. part = "g";
  163. CPPUNIT_ASSERT(util::endsWith(target, part));
  164. target = "abdefg";
  165. part = "eg";
  166. CPPUNIT_ASSERT(!util::endsWith(target, part));
  167. target = "g";
  168. part = "eg";
  169. CPPUNIT_ASSERT(!util::endsWith(target, part));
  170. target = "g";
  171. part = "g";
  172. CPPUNIT_ASSERT(util::endsWith(target, part));
  173. target = "g";
  174. part = "";
  175. CPPUNIT_ASSERT(util::endsWith(target, part));
  176. target = "";
  177. part = "";
  178. CPPUNIT_ASSERT(util::endsWith(target, part));
  179. target = "";
  180. part = "g";
  181. CPPUNIT_ASSERT(!util::endsWith(target, part));
  182. }
  183. void UtilTest::testReplace() {
  184. CPPUNIT_ASSERT_EQUAL(std::string("abc\n"), util::replace("abc\r\n", "\r", ""));
  185. CPPUNIT_ASSERT_EQUAL(std::string("abc"), util::replace("abc\r\n", "\r\n", ""));
  186. CPPUNIT_ASSERT_EQUAL(std::string(""), util::replace("", "\r\n", ""));
  187. CPPUNIT_ASSERT_EQUAL(std::string("abc"), util::replace("abc", "", "a"));
  188. CPPUNIT_ASSERT_EQUAL(std::string("xbc"), util::replace("abc", "a", "x"));
  189. }
  190. void UtilTest::testStartsWith() {
  191. std::string target;
  192. std::string part;
  193. target = "abcdefg";
  194. part = "abc";
  195. CPPUNIT_ASSERT(util::startsWith(target, part));
  196. target = "abcdefg";
  197. part = "abx";
  198. CPPUNIT_ASSERT(!util::startsWith(target, part));
  199. target = "abcdefg";
  200. part = "bcd";
  201. CPPUNIT_ASSERT(!util::startsWith(target, part));
  202. target = "";
  203. part = "a";
  204. CPPUNIT_ASSERT(!util::startsWith(target, part));
  205. target = "";
  206. part = "";
  207. CPPUNIT_ASSERT(util::startsWith(target, part));
  208. target = "a";
  209. part = "";
  210. CPPUNIT_ASSERT(util::startsWith(target, part));
  211. target = "a";
  212. part = "a";
  213. CPPUNIT_ASSERT(util::startsWith(target, part));
  214. }
  215. void UtilTest::testGetContentDispositionFilename() {
  216. std::string h1 = "attachment; filename=\"aria2.tar.bz2\"";
  217. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h1));
  218. std::string h2 = "attachment; filename=\"\"";
  219. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h2));
  220. std::string h3 = "attachment; filename=\"";
  221. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h3));
  222. std::string h4 = "attachment;";
  223. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h4));
  224. std::string h5 = "attachment; filename=aria2.tar.bz2";
  225. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h5));
  226. std::string h6 = "attachment; filename='aria2.tar.bz2'";
  227. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h6));
  228. std::string h7 = "attachment; filename='aria2.tar.bz2";
  229. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h7));
  230. std::string h8 = "attachment; filename=aria2.tar.bz2; creation-date=20 Jun 2007 00:00:00 GMT";
  231. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"), util::getContentDispositionFilename(h8));
  232. std::string h9 = "attachment; filename=\"aria2.tar.bz2; creation-date=20 Jun 2007 00:00:00 GMT\"";
  233. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2; creation-date=20 Jun 2007 00:00:00 GMT"), util::getContentDispositionFilename(h9));
  234. std::string h10 = "attachment; filename=";
  235. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h10));
  236. std::string h11 = "attachment; filename=;";
  237. CPPUNIT_ASSERT_EQUAL(std::string(""), util::getContentDispositionFilename(h11));
  238. std::string filenameWithDir = "attachment; filename=dir/file";
  239. CPPUNIT_ASSERT_EQUAL(std::string("file"),
  240. util::getContentDispositionFilename(filenameWithDir));
  241. std::string parentDir = "attachment; filename=..";
  242. CPPUNIT_ASSERT_EQUAL(std::string(),
  243. util::getContentDispositionFilename(parentDir));
  244. std::string currentDir = "attachment; filename=.";
  245. CPPUNIT_ASSERT_EQUAL(std::string(),
  246. util::getContentDispositionFilename(currentDir));
  247. }
  248. class Printer {
  249. public:
  250. template<class T>
  251. void operator()(T t) {
  252. std::cerr << t << ", ";
  253. }
  254. };
  255. void UtilTest::testRandomAlpha() {
  256. SharedHandle<Randomizer> rand(new FixedNumberRandomizer());
  257. std::string s = util::randomAlpha(8, rand);
  258. CPPUNIT_ASSERT_EQUAL(std::string("AAAAAAAA"), s);
  259. }
  260. void UtilTest::testToUpper() {
  261. std::string src = "608cabc0f2fa18c260cafd974516865c772363d5";
  262. std::string upp = "608CABC0F2FA18C260CAFD974516865C772363D5";
  263. CPPUNIT_ASSERT_EQUAL(upp, util::toUpper(src));
  264. }
  265. void UtilTest::testToLower() {
  266. std::string src = "608CABC0F2FA18C260CAFD974516865C772363D5";
  267. std::string upp = "608cabc0f2fa18c260cafd974516865c772363d5";
  268. CPPUNIT_ASSERT_EQUAL(upp, util::toLower(src));
  269. }
  270. void UtilTest::testUrldecode() {
  271. std::string src = "http://aria2.sourceforge.net/aria2%200.7.0%20docs.html";
  272. CPPUNIT_ASSERT_EQUAL(std::string("http://aria2.sourceforge.net/aria2 0.7.0 docs.html"),
  273. util::urldecode(src));
  274. std::string src2 = "aria2+aria2";
  275. CPPUNIT_ASSERT_EQUAL(std::string("aria2+aria2"), util::urldecode(src2));
  276. std::string src3 = "%5t%20";
  277. CPPUNIT_ASSERT_EQUAL(std::string("%5t "), util::urldecode(src3));
  278. std::string src4 = "%";
  279. CPPUNIT_ASSERT_EQUAL(std::string("%"), util::urldecode(src4));
  280. std::string src5 = "%3";
  281. CPPUNIT_ASSERT_EQUAL(std::string("%3"), util::urldecode(src5));
  282. std::string src6 = "%2f";
  283. CPPUNIT_ASSERT_EQUAL(std::string("/"), util::urldecode(src6));
  284. }
  285. void UtilTest::testGetRealSize()
  286. {
  287. CPPUNIT_ASSERT_EQUAL((int64_t)4294967296LL, util::getRealSize("4096M"));
  288. CPPUNIT_ASSERT_EQUAL((int64_t)1024, util::getRealSize("1K"));
  289. try {
  290. util::getRealSize("");
  291. CPPUNIT_FAIL("exception must be thrown.");
  292. } catch(Exception& e) {
  293. std::cerr << e.stackTrace();
  294. }
  295. try {
  296. util::getRealSize("foo");
  297. CPPUNIT_FAIL("exception must be thrown.");
  298. } catch(Exception& e) {
  299. std::cerr << e.stackTrace();
  300. }
  301. try {
  302. util::getRealSize("-1");
  303. CPPUNIT_FAIL("exception must be thrown.");
  304. } catch(Exception& e) {
  305. std::cerr << e.stackTrace();
  306. }
  307. try {
  308. util::getRealSize("9223372036854775807K");
  309. CPPUNIT_FAIL("exception must be thrown.");
  310. } catch(Exception& e) {
  311. std::cerr << e.stackTrace();
  312. }
  313. try {
  314. util::getRealSize("9223372036854775807M");
  315. CPPUNIT_FAIL("exception must be thrown.");
  316. } catch(Exception& e) {
  317. std::cerr << e.stackTrace();
  318. }
  319. }
  320. void UtilTest::testAbbrevSize()
  321. {
  322. CPPUNIT_ASSERT_EQUAL(std::string("4,096.0Mi"), util::abbrevSize(4294967296LL));
  323. CPPUNIT_ASSERT_EQUAL(std::string("1.0Ki"), util::abbrevSize(1024));
  324. CPPUNIT_ASSERT_EQUAL(std::string("1,023"), util::abbrevSize(1023));
  325. CPPUNIT_ASSERT_EQUAL(std::string("0"), util::abbrevSize(0));
  326. CPPUNIT_ASSERT_EQUAL(std::string("1.1Ki"), util::abbrevSize(1127));
  327. CPPUNIT_ASSERT_EQUAL(std::string("1.5Mi"), util::abbrevSize(1572864));
  328. }
  329. void UtilTest::testToStream()
  330. {
  331. std::ostringstream os;
  332. SharedHandle<FileEntry> f1(new FileEntry("aria2.tar.bz2", 12300, 0));
  333. SharedHandle<FileEntry> f2(new FileEntry("aria2.txt", 556, 0));
  334. std::deque<SharedHandle<FileEntry> > entries;
  335. entries.push_back(f1);
  336. entries.push_back(f2);
  337. util::toStream(entries.begin(), entries.end(), os);
  338. CPPUNIT_ASSERT_EQUAL(
  339. std::string("Files:\n"
  340. "idx|path/length\n"
  341. "===+===========================================================================\n"
  342. " 1|aria2.tar.bz2\n"
  343. " |12.0KiB (12,300)\n"
  344. "---+---------------------------------------------------------------------------\n"
  345. " 2|aria2.txt\n"
  346. " |556B (556)\n"
  347. "---+---------------------------------------------------------------------------\n"),
  348. os.str());
  349. }
  350. void UtilTest::testIsNumber()
  351. {
  352. CPPUNIT_ASSERT_EQUAL(true, util::isNumber("000"));
  353. CPPUNIT_ASSERT_EQUAL(false, util::isNumber("a"));
  354. CPPUNIT_ASSERT_EQUAL(false, util::isNumber("0a"));
  355. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(""));
  356. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(" "));
  357. }
  358. void UtilTest::testIsLowercase()
  359. {
  360. CPPUNIT_ASSERT_EQUAL(true, util::isLowercase("alpha"));
  361. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase("Alpha"));
  362. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase("1alpha"));
  363. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(""));
  364. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(" "));
  365. }
  366. void UtilTest::testIsUppercase()
  367. {
  368. CPPUNIT_ASSERT_EQUAL(true, util::isUppercase("ALPHA"));
  369. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase("Alpha"));
  370. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase("1ALPHA"));
  371. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(""));
  372. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(" "));
  373. }
  374. void UtilTest::testAlphaToNum()
  375. {
  376. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum("a"));
  377. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum("aa"));
  378. CPPUNIT_ASSERT_EQUAL(1U, util::alphaToNum("b"));
  379. CPPUNIT_ASSERT_EQUAL(675U, util::alphaToNum("zz")); // 25*26+25
  380. CPPUNIT_ASSERT_EQUAL(675U, util::alphaToNum("ZZ")); // 25*26+25
  381. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum(""));
  382. CPPUNIT_ASSERT_EQUAL(4294967295U, util::alphaToNum("NXMRLXV"));
  383. CPPUNIT_ASSERT_EQUAL(0U, util::alphaToNum("NXMRLXW")); // uint32_t overflow
  384. }
  385. void UtilTest::testMkdirs()
  386. {
  387. std::string dir = "/tmp/aria2-UtilTest-testMkdirs";
  388. File d(dir);
  389. if(d.exists()) {
  390. CPPUNIT_ASSERT(d.remove());
  391. }
  392. CPPUNIT_ASSERT(!d.exists());
  393. util::mkdirs(dir);
  394. CPPUNIT_ASSERT(d.isDir());
  395. std::string file = "./UtilTest.cc";
  396. File f(file);
  397. CPPUNIT_ASSERT(f.isFile());
  398. try {
  399. util::mkdirs(file);
  400. CPPUNIT_FAIL("exception must be thrown.");
  401. } catch(DlAbortEx& ex) {
  402. std::cerr << ex.stackTrace() << std::endl;
  403. }
  404. }
  405. void UtilTest::testConvertBitfield()
  406. {
  407. BitfieldMan srcBitfield(384*1024, 256*1024*256+1);
  408. BitfieldMan destBitfield(512*1024, srcBitfield.getTotalLength());
  409. srcBitfield.setAllBit();
  410. srcBitfield.unsetBit(2);// <- range [768, 1152)
  411. // which corresponds to the index [1,2] in destBitfield
  412. util::convertBitfield(&destBitfield, &srcBitfield);
  413. CPPUNIT_ASSERT_EQUAL(std::string("9fffffffffffffffffffffffffffffff80"),
  414. util::toHex(destBitfield.getBitfield(),
  415. destBitfield.getBitfieldLength()));
  416. }
  417. void UtilTest::testParseIntRange()
  418. {
  419. IntSequence seq = util::parseIntRange("1,3-8,10");
  420. CPPUNIT_ASSERT(seq.hasNext());
  421. CPPUNIT_ASSERT_EQUAL((int32_t)1, seq.next());
  422. CPPUNIT_ASSERT(seq.hasNext());
  423. CPPUNIT_ASSERT_EQUAL((int32_t)3, seq.next());
  424. CPPUNIT_ASSERT(seq.hasNext());
  425. CPPUNIT_ASSERT_EQUAL((int32_t)4, seq.next());
  426. CPPUNIT_ASSERT(seq.hasNext());
  427. CPPUNIT_ASSERT_EQUAL((int32_t)5, seq.next());
  428. CPPUNIT_ASSERT(seq.hasNext());
  429. CPPUNIT_ASSERT_EQUAL((int32_t)6, seq.next());
  430. CPPUNIT_ASSERT(seq.hasNext());
  431. CPPUNIT_ASSERT_EQUAL((int32_t)7, seq.next());
  432. CPPUNIT_ASSERT(seq.hasNext());
  433. CPPUNIT_ASSERT_EQUAL((int32_t)8, seq.next());
  434. CPPUNIT_ASSERT(seq.hasNext());
  435. CPPUNIT_ASSERT_EQUAL((int32_t)10, seq.next());
  436. CPPUNIT_ASSERT(!seq.hasNext());
  437. CPPUNIT_ASSERT_EQUAL((int32_t)0, seq.next());
  438. }
  439. void UtilTest::testParseIntRange_invalidRange()
  440. {
  441. try {
  442. IntSequence seq = util::parseIntRange("-1");
  443. CPPUNIT_FAIL("exception must be thrown.");
  444. } catch(Exception& e) {
  445. std::cerr << e.stackTrace();
  446. }
  447. try {
  448. IntSequence seq = util::parseIntRange("2147483648");
  449. CPPUNIT_FAIL("exception must be thrown.");
  450. } catch(Exception& e) {
  451. std::cerr << e.stackTrace();
  452. }
  453. try {
  454. IntSequence seq = util::parseIntRange("2147483647-2147483648");
  455. CPPUNIT_FAIL("exception must be thrown.");
  456. } catch(Exception& e) {
  457. std::cerr << e.stackTrace();
  458. }
  459. try {
  460. IntSequence seq = util::parseIntRange("1-2x");
  461. CPPUNIT_FAIL("exception must be thrown.");
  462. } catch(Exception& e) {
  463. std::cerr << e.stackTrace();
  464. }
  465. try {
  466. IntSequence seq = util::parseIntRange("3x-4");
  467. CPPUNIT_FAIL("exception must be thrown.");
  468. } catch(Exception& e) {
  469. std::cerr << e.stackTrace();
  470. }
  471. }
  472. void UtilTest::testParseInt()
  473. {
  474. CPPUNIT_ASSERT_EQUAL(-1, util::parseInt(" -1 "));
  475. CPPUNIT_ASSERT_EQUAL(2147483647, util::parseInt("2147483647"));
  476. try {
  477. util::parseInt("2147483648");
  478. CPPUNIT_FAIL("exception must be thrown.");
  479. } catch(Exception& e) {
  480. std::cerr << e.stackTrace();
  481. }
  482. try {
  483. util::parseInt("-2147483649");
  484. CPPUNIT_FAIL("exception must be thrown.");
  485. } catch(Exception& e) {
  486. std::cerr << e.stackTrace();
  487. }
  488. try {
  489. util::parseInt("12x");
  490. CPPUNIT_FAIL("exception must be thrown.");
  491. } catch(Exception& e) {
  492. std::cerr << e.stackTrace();
  493. }
  494. try {
  495. util::parseInt("");
  496. CPPUNIT_FAIL("exception must be thrown.");
  497. } catch(Exception& e) {
  498. std::cerr << e.stackTrace();
  499. }
  500. }
  501. void UtilTest::testParseUInt()
  502. {
  503. CPPUNIT_ASSERT_EQUAL(4294967295U, util::parseUInt(" 4294967295 "));
  504. try {
  505. util::parseUInt("-1");
  506. CPPUNIT_FAIL("exception must be thrown.");
  507. } catch(Exception& e) {
  508. std::cerr << e.stackTrace();
  509. }
  510. try {
  511. util::parseUInt("4294967296");
  512. CPPUNIT_FAIL("exception must be thrown.");
  513. } catch(Exception& e) {
  514. std::cerr << e.stackTrace();
  515. }
  516. }
  517. void UtilTest::testParseLLInt()
  518. {
  519. CPPUNIT_ASSERT_EQUAL((int64_t)-1LL, util::parseLLInt(" -1 "));
  520. CPPUNIT_ASSERT_EQUAL((int64_t)9223372036854775807LL,
  521. util::parseLLInt("9223372036854775807"));
  522. try {
  523. util::parseLLInt("9223372036854775808");
  524. CPPUNIT_FAIL("exception must be thrown.");
  525. } catch(Exception& e) {
  526. std::cerr << e.stackTrace();
  527. }
  528. try {
  529. util::parseLLInt("-9223372036854775809");
  530. CPPUNIT_FAIL("exception must be thrown.");
  531. } catch(Exception& e) {
  532. std::cerr << e.stackTrace();
  533. }
  534. try {
  535. util::parseLLInt("12x");
  536. CPPUNIT_FAIL("exception must be thrown.");
  537. } catch(Exception& e) {
  538. std::cerr << e.stackTrace();
  539. }
  540. try {
  541. util::parseLLInt("");
  542. CPPUNIT_FAIL("exception must be thrown.");
  543. } catch(Exception& e) {
  544. std::cerr << e.stackTrace();
  545. }
  546. }
  547. void UtilTest::testParseULLInt()
  548. {
  549. CPPUNIT_ASSERT_EQUAL((uint64_t)18446744073709551615ULL,
  550. util::parseULLInt("18446744073709551615"));
  551. try {
  552. util::parseUInt("-1");
  553. CPPUNIT_FAIL("exception must be thrown.");
  554. } catch(Exception& e) {
  555. std::cerr << e.stackTrace();
  556. }
  557. try {
  558. util::parseLLInt("18446744073709551616");
  559. CPPUNIT_FAIL("exception must be thrown.");
  560. } catch(Exception& e) {
  561. std::cerr << e.stackTrace();
  562. }
  563. }
  564. void UtilTest::testToString_binaryStream()
  565. {
  566. SharedHandle<DiskWriter> dw(new ByteArrayDiskWriter());
  567. std::string data(16*1024+256, 'a');
  568. dw->initAndOpenFile();
  569. dw->writeData((const unsigned char*)data.c_str(), data.size(), 0);
  570. std::string readData = util::toString(dw);
  571. CPPUNIT_ASSERT_EQUAL(data, readData);
  572. }
  573. void UtilTest::testItos()
  574. {
  575. {
  576. int i = 0;
  577. CPPUNIT_ASSERT_EQUAL(std::string("0"), util::itos(i));
  578. }
  579. {
  580. int i = 100;
  581. CPPUNIT_ASSERT_EQUAL(std::string("100"), util::itos(i, true));
  582. }
  583. {
  584. int i = 100;
  585. CPPUNIT_ASSERT_EQUAL(std::string("100"), util::itos(i));
  586. }
  587. {
  588. int i = 12345;
  589. CPPUNIT_ASSERT_EQUAL(std::string("12,345"), util::itos(i, true));
  590. }
  591. {
  592. int i = 12345;
  593. CPPUNIT_ASSERT_EQUAL(std::string("12345"), util::itos(i));
  594. }
  595. {
  596. int i = -12345;
  597. CPPUNIT_ASSERT_EQUAL(std::string("-12,345"), util::itos(i, true));
  598. }
  599. {
  600. int64_t i = INT64_MAX;
  601. CPPUNIT_ASSERT_EQUAL(std::string("9,223,372,036,854,775,807"), util::itos(i, true));
  602. }
  603. }
  604. void UtilTest::testUitos()
  605. {
  606. {
  607. uint16_t i = 12345;
  608. CPPUNIT_ASSERT_EQUAL(std::string("12345"), util::uitos(i));
  609. }
  610. {
  611. int16_t i = -12345;
  612. CPPUNIT_ASSERT_EQUAL(std::string("/.-,+"), util::uitos(i));
  613. }
  614. }
  615. void UtilTest::testNtoh64()
  616. {
  617. uint64_t x = 0xff00ff00ee00ee00LL;
  618. #ifdef WORDS_BIGENDIAN
  619. CPPUNIT_ASSERT_EQUAL(x, ntoh64(x));
  620. CPPUNIT_ASSERT_EQUAL(x, hton64(x));
  621. #else // !WORDS_BIGENDIAN
  622. uint64_t y = 0x00ee00ee00ff00ffLL;
  623. CPPUNIT_ASSERT_EQUAL(y, ntoh64(x));
  624. CPPUNIT_ASSERT_EQUAL(x, hton64(y));
  625. #endif // !WORDS_BIGENDIAN
  626. }
  627. void UtilTest::testUrlencode()
  628. {
  629. CPPUNIT_ASSERT_EQUAL
  630. (std::string("%3A%2F%3F%23%5B%5D%40%21%25%26%27%28%29%2A%2B%2C%3B%3D"),
  631. util::urlencode(":/?#[]@!%&'()*+,;="));
  632. std::string unreserved =
  633. "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  634. "abcdefghijklmnopqrstuvwxyz"
  635. "0123456789"
  636. "-._~";
  637. CPPUNIT_ASSERT_EQUAL(unreserved, util::urlencode(unreserved));
  638. CPPUNIT_ASSERT_EQUAL(std::string("1%5EA%20"), util::urlencode("1^A "));
  639. }
  640. void UtilTest::testHtmlEscape()
  641. {
  642. CPPUNIT_ASSERT_EQUAL(std::string("aria2&lt;&gt;&quot;&#39;util"),
  643. util::htmlEscape("aria2<>\"'util"));
  644. }
  645. void UtilTest::testJoinPath()
  646. {
  647. const std::string dir1dir2file[] = { "dir1", "dir2", "file" };
  648. CPPUNIT_ASSERT_EQUAL
  649. (std::string("dir1/dir2/file"),
  650. util::joinPath(&dir1dir2file[0],
  651. &dir1dir2file[arrayLength(dir1dir2file)]));
  652. const std::string dirparentfile[] = { "dir", "..", "file" };
  653. CPPUNIT_ASSERT_EQUAL
  654. (std::string("file"),
  655. util::joinPath(&dirparentfile[0],
  656. &dirparentfile[arrayLength(dirparentfile)]));
  657. const std::string dirparentparentfile[] = { "dir", "..", "..", "file" };
  658. CPPUNIT_ASSERT_EQUAL
  659. (std::string("file"),
  660. util::joinPath(&dirparentparentfile[0],
  661. &dirparentparentfile[arrayLength(dirparentparentfile)]));
  662. const std::string dirdotfile[] = { "dir", ".", "file" };
  663. CPPUNIT_ASSERT_EQUAL(std::string("dir/file"),
  664. util::joinPath(&dirdotfile[0],
  665. &dirdotfile[arrayLength(dirdotfile)]));
  666. const std::string empty[] = {};
  667. CPPUNIT_ASSERT_EQUAL(std::string(""), util::joinPath(&empty[0], &empty[0]));
  668. const std::string parentdot[] = { "..", "." };
  669. CPPUNIT_ASSERT_EQUAL(std::string(""),
  670. util::joinPath(&parentdot[0],
  671. &parentdot[arrayLength(parentdot)]));
  672. }
  673. void UtilTest::testParseIndexPath()
  674. {
  675. std::map<size_t, std::string>::value_type p = util::parseIndexPath("1=foo");
  676. CPPUNIT_ASSERT_EQUAL((size_t)1, p.first);
  677. CPPUNIT_ASSERT_EQUAL(std::string("foo"), p.second);
  678. try {
  679. util::parseIndexPath("1X=foo");
  680. CPPUNIT_FAIL("exception must be thrown.");
  681. } catch(Exception& e) {
  682. // success
  683. }
  684. try {
  685. util::parseIndexPath("1=");
  686. CPPUNIT_FAIL("exception must be thrown.");
  687. } catch(Exception& e) {
  688. // success
  689. }
  690. }
  691. void UtilTest::testCreateIndexPathMap()
  692. {
  693. std::stringstream in
  694. ("1=/tmp/myfile\n"
  695. "100=/myhome/mypicture.png\n");
  696. std::map<size_t, std::string> m = util::createIndexPathMap(in);
  697. CPPUNIT_ASSERT_EQUAL((size_t)2, m.size());
  698. CPPUNIT_ASSERT(m.find(1) != m.end());
  699. CPPUNIT_ASSERT_EQUAL(std::string("/tmp/myfile"), m[1]);
  700. CPPUNIT_ASSERT(m.find(100) != m.end());
  701. CPPUNIT_ASSERT_EQUAL(std::string("/myhome/mypicture.png"), m[100]);
  702. }
  703. void UtilTest::testGenerateRandomData()
  704. {
  705. unsigned char data1[20];
  706. util::generateRandomData(data1, sizeof(data1));
  707. unsigned char data2[20];
  708. util::generateRandomData(data2, sizeof(data2));
  709. CPPUNIT_ASSERT(memcmp(data1, data2, sizeof(data1)) != 0);
  710. }
  711. void UtilTest::testFromHex()
  712. {
  713. std::string src;
  714. std::string dest;
  715. src = "0011fF";
  716. dest = util::fromHex(src);
  717. CPPUNIT_ASSERT_EQUAL((size_t)3, dest.size());
  718. CPPUNIT_ASSERT_EQUAL((char)0x00, dest[0]);
  719. CPPUNIT_ASSERT_EQUAL((char)0x11, dest[1]);
  720. CPPUNIT_ASSERT_EQUAL((char)0xff, dest[2]);
  721. src = "0011f";
  722. CPPUNIT_ASSERT(util::fromHex(src).empty());
  723. src = "001g";
  724. CPPUNIT_ASSERT(util::fromHex(src).empty());
  725. }
  726. void UtilTest::testParsePrioritizePieceRange()
  727. {
  728. // piece index
  729. // 0 1 2 3 4 5 6 7
  730. // | | | |
  731. // file1 | | |
  732. // | | |
  733. // file2 | |
  734. // file3 |
  735. // | |
  736. // file4 |
  737. size_t pieceLength = 1024;
  738. std::vector<SharedHandle<FileEntry> > entries(4, SharedHandle<FileEntry>());
  739. entries[0].reset(new FileEntry("file1", 1024, 0));
  740. entries[1].reset(new FileEntry("file2",2560,entries[0]->getLastOffset()));
  741. entries[2].reset(new FileEntry("file3",0,entries[1]->getLastOffset()));
  742. entries[3].reset(new FileEntry("file4",3584,entries[2]->getLastOffset()));
  743. std::vector<size_t> result;
  744. util::parsePrioritizePieceRange(result, "head=1", entries, pieceLength);
  745. CPPUNIT_ASSERT_EQUAL((size_t)3, result.size());
  746. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  747. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  748. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  749. result.clear();
  750. util::parsePrioritizePieceRange(result, "tail=1", entries, pieceLength);
  751. CPPUNIT_ASSERT_EQUAL((size_t)3, result.size());
  752. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  753. CPPUNIT_ASSERT_EQUAL((size_t)3, result[1]);
  754. CPPUNIT_ASSERT_EQUAL((size_t)6, result[2]);
  755. result.clear();
  756. util::parsePrioritizePieceRange(result, "head=1K", entries, pieceLength);
  757. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  758. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  759. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  760. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  761. CPPUNIT_ASSERT_EQUAL((size_t)4, result[3]);
  762. result.clear();
  763. util::parsePrioritizePieceRange(result, "head", entries, pieceLength, 1024);
  764. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  765. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  766. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  767. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  768. CPPUNIT_ASSERT_EQUAL((size_t)4, result[3]);
  769. result.clear();
  770. util::parsePrioritizePieceRange(result, "tail=1K", entries, pieceLength);
  771. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  772. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  773. CPPUNIT_ASSERT_EQUAL((size_t)2, result[1]);
  774. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  775. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  776. result.clear();
  777. util::parsePrioritizePieceRange(result, "tail", entries, pieceLength, 1024);
  778. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  779. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  780. CPPUNIT_ASSERT_EQUAL((size_t)2, result[1]);
  781. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  782. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  783. result.clear();
  784. util::parsePrioritizePieceRange(result, "head=1,tail=1", entries, pieceLength);
  785. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  786. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  787. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  788. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  789. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  790. result.clear();
  791. util::parsePrioritizePieceRange(result, "head=300M,tail=300M",
  792. entries, pieceLength);
  793. CPPUNIT_ASSERT_EQUAL((size_t)7, result.size());
  794. for(size_t i = 0; i < 7; ++i) {
  795. CPPUNIT_ASSERT_EQUAL(i, result[i]);
  796. }
  797. result.clear();
  798. util::parsePrioritizePieceRange(result, "", entries, pieceLength);
  799. CPPUNIT_ASSERT(result.empty());
  800. }
  801. void UtilTest::testApplyDir()
  802. {
  803. CPPUNIT_ASSERT_EQUAL(std::string("./pred"), util::applyDir("", "pred"));
  804. CPPUNIT_ASSERT_EQUAL(std::string("/pred"), util::applyDir("/", "pred"));
  805. CPPUNIT_ASSERT_EQUAL(std::string("./pred"), util::applyDir(".", "pred"));
  806. CPPUNIT_ASSERT_EQUAL(std::string("/dl/pred"), util::applyDir("/dl", "pred"));
  807. }
  808. } // namespace aria2