UtilTest.cc 84 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307
  1. #include "util.h"
  2. #include <cmath>
  3. #include <cstring>
  4. #include <string>
  5. #include <iostream>
  6. #include <cppunit/extensions/HelperMacros.h>
  7. #include "FixedNumberRandomizer.h"
  8. #include "DlAbortEx.h"
  9. #include "BitfieldMan.h"
  10. #include "ByteArrayDiskWriter.h"
  11. #include "FileEntry.h"
  12. #include "File.h"
  13. #include "array_fun.h"
  14. #include "BufferedFile.h"
  15. #include "TestUtil.h"
  16. #include "SocketCore.h"
  17. namespace aria2 {
  18. class UtilTest:public CppUnit::TestFixture {
  19. CPPUNIT_TEST_SUITE(UtilTest);
  20. CPPUNIT_TEST(testStrip);
  21. CPPUNIT_TEST(testStripIter);
  22. CPPUNIT_TEST(testLstripIter);
  23. CPPUNIT_TEST(testLstripIter_char);
  24. CPPUNIT_TEST(testDivide);
  25. CPPUNIT_TEST(testSplit);
  26. CPPUNIT_TEST(testSplitIter);
  27. CPPUNIT_TEST(testSplitIterM);
  28. CPPUNIT_TEST(testStreq);
  29. CPPUNIT_TEST(testStrieq);
  30. CPPUNIT_TEST(testStrifind);
  31. CPPUNIT_TEST(testEndsWith);
  32. CPPUNIT_TEST(testIendsWith);
  33. CPPUNIT_TEST(testReplace);
  34. CPPUNIT_TEST(testStartsWith);
  35. CPPUNIT_TEST(testIstartsWith);
  36. // may be moved to other helper class in the future.
  37. CPPUNIT_TEST(testGetContentDispositionFilename);
  38. CPPUNIT_TEST(testParseContentDisposition);
  39. CPPUNIT_TEST(testToUpper);
  40. CPPUNIT_TEST(testToLower);
  41. CPPUNIT_TEST(testUppercase);
  42. CPPUNIT_TEST(testLowercase);
  43. CPPUNIT_TEST(testPercentDecode);
  44. CPPUNIT_TEST(testGetRealSize);
  45. CPPUNIT_TEST(testAbbrevSize);
  46. CPPUNIT_TEST(testToStream);
  47. CPPUNIT_TEST(testIsNumber);
  48. CPPUNIT_TEST(testIsLowercase);
  49. CPPUNIT_TEST(testIsUppercase);
  50. CPPUNIT_TEST(testMkdirs);
  51. CPPUNIT_TEST(testConvertBitfield);
  52. CPPUNIT_TEST(testParseIntSegments);
  53. CPPUNIT_TEST(testParseIntSegments_invalidRange);
  54. CPPUNIT_TEST(testParseIntNoThrow);
  55. CPPUNIT_TEST(testParseUIntNoThrow);
  56. CPPUNIT_TEST(testParseLLIntNoThrow);
  57. CPPUNIT_TEST(testToString_binaryStream);
  58. CPPUNIT_TEST(testItos);
  59. CPPUNIT_TEST(testUitos);
  60. CPPUNIT_TEST(testNtoh64);
  61. CPPUNIT_TEST(testPercentEncode);
  62. CPPUNIT_TEST(testPercentEncodeMini);
  63. CPPUNIT_TEST(testHtmlEscape);
  64. CPPUNIT_TEST(testJoinPath);
  65. CPPUNIT_TEST(testParseIndexPath);
  66. CPPUNIT_TEST(testCreateIndexPaths);
  67. CPPUNIT_TEST(testGenerateRandomData);
  68. CPPUNIT_TEST(testFromHex);
  69. CPPUNIT_TEST(testParsePrioritizePieceRange);
  70. CPPUNIT_TEST(testApplyDir);
  71. CPPUNIT_TEST(testFixTaintedBasename);
  72. CPPUNIT_TEST(testIsNumericHost);
  73. CPPUNIT_TEST(testDetectDirTraversal);
  74. CPPUNIT_TEST(testEscapePath);
  75. CPPUNIT_TEST(testInSameCidrBlock);
  76. CPPUNIT_TEST(testIsUtf8String);
  77. CPPUNIT_TEST(testNextParam);
  78. CPPUNIT_TEST(testNoProxyDomainMatch);
  79. CPPUNIT_TEST(testInPrivateAddress);
  80. CPPUNIT_TEST(testSecfmt);
  81. CPPUNIT_TEST(testTlsHostnameMatch);
  82. CPPUNIT_TEST_SUITE_END();
  83. private:
  84. public:
  85. void setUp() {
  86. }
  87. void testStrip();
  88. void testStripIter();
  89. void testLstripIter();
  90. void testLstripIter_char();
  91. void testDivide();
  92. void testSplit();
  93. void testSplitIter();
  94. void testSplitIterM();
  95. void testStreq();
  96. void testStrieq();
  97. void testStrifind();
  98. void testEndsWith();
  99. void testIendsWith();
  100. void testReplace();
  101. void testStartsWith();
  102. void testIstartsWith();
  103. // may be moved to other helper class in the future.
  104. void testGetContentDispositionFilename();
  105. void testParseContentDisposition();
  106. void testToUpper();
  107. void testToLower();
  108. void testUppercase();
  109. void testLowercase();
  110. void testPercentDecode();
  111. void testGetRealSize();
  112. void testAbbrevSize();
  113. void testToStream();
  114. void testIsNumber();
  115. void testIsLowercase();
  116. void testIsUppercase();
  117. void testMkdirs();
  118. void testConvertBitfield();
  119. void testParseIntSegments();
  120. void testParseIntSegments_invalidRange();
  121. void testParseIntNoThrow();
  122. void testParseUIntNoThrow();
  123. void testParseLLIntNoThrow();
  124. void testToString_binaryStream();
  125. void testItos();
  126. void testUitos();
  127. void testNtoh64();
  128. void testPercentEncode();
  129. void testPercentEncodeMini();
  130. void testHtmlEscape();
  131. void testJoinPath();
  132. void testParseIndexPath();
  133. void testCreateIndexPaths();
  134. void testGenerateRandomData();
  135. void testFromHex();
  136. void testParsePrioritizePieceRange();
  137. void testApplyDir();
  138. void testFixTaintedBasename();
  139. void testIsNumericHost();
  140. void testDetectDirTraversal();
  141. void testEscapePath();
  142. void testInSameCidrBlock();
  143. void testIsUtf8String();
  144. void testNextParam();
  145. void testNoProxyDomainMatch();
  146. void testInPrivateAddress();
  147. void testSecfmt();
  148. void testTlsHostnameMatch();
  149. };
  150. CPPUNIT_TEST_SUITE_REGISTRATION( UtilTest );
  151. void UtilTest::testStrip()
  152. {
  153. std::string str1 = "aria2";
  154. CPPUNIT_ASSERT_EQUAL(str1, util::strip("aria2"));
  155. CPPUNIT_ASSERT_EQUAL(str1, util::strip(" aria2"));
  156. CPPUNIT_ASSERT_EQUAL(str1, util::strip("aria2 "));
  157. CPPUNIT_ASSERT_EQUAL(str1, util::strip(" aria2 "));
  158. CPPUNIT_ASSERT_EQUAL(str1, util::strip(" aria2 "));
  159. std::string str2 = "aria2 debut";
  160. CPPUNIT_ASSERT_EQUAL(str2, util::strip("aria2 debut"));
  161. CPPUNIT_ASSERT_EQUAL(str2, util::strip(" aria2 debut "));
  162. std::string str3 = "";
  163. CPPUNIT_ASSERT_EQUAL(str3, util::strip(""));
  164. CPPUNIT_ASSERT_EQUAL(str3, util::strip(" "));
  165. CPPUNIT_ASSERT_EQUAL(str3, util::strip(" "));
  166. std::string str4 = "A";
  167. CPPUNIT_ASSERT_EQUAL(str4, util::strip("A"));
  168. CPPUNIT_ASSERT_EQUAL(str4, util::strip(" A "));
  169. CPPUNIT_ASSERT_EQUAL(str4, util::strip(" A "));
  170. }
  171. void UtilTest::testStripIter()
  172. {
  173. Scip p;
  174. std::string str1 = "aria2";
  175. std::string s = "aria2";
  176. p = util::stripIter(s.begin(), s.end());
  177. CPPUNIT_ASSERT_EQUAL(str1, std::string(p.first, p.second));
  178. s = " aria2";
  179. p = util::stripIter(s.begin(), s.end());
  180. CPPUNIT_ASSERT_EQUAL(str1, std::string(p.first, p.second));
  181. s = "aria2 ";
  182. p = util::stripIter(s.begin(), s.end());
  183. CPPUNIT_ASSERT_EQUAL(str1, std::string(p.first, p.second));
  184. s = " aria2 ";
  185. p = util::stripIter(s.begin(), s.end());
  186. CPPUNIT_ASSERT_EQUAL(str1, std::string(p.first, p.second));
  187. s = " aria2 ";
  188. p = util::stripIter(s.begin(), s.end());
  189. CPPUNIT_ASSERT_EQUAL(str1, std::string(p.first, p.second));
  190. std::string str2 = "aria2 debut";
  191. s = "aria2 debut";
  192. p = util::stripIter(s.begin(), s.end());
  193. CPPUNIT_ASSERT_EQUAL(str2, std::string(p.first, p.second));
  194. s = " aria2 debut ";
  195. p = util::stripIter(s.begin(), s.end());
  196. CPPUNIT_ASSERT_EQUAL(str2, std::string(p.first, p.second));
  197. std::string str3 = "";
  198. s = "";
  199. p = util::stripIter(s.begin(), s.end());
  200. CPPUNIT_ASSERT_EQUAL(str3, std::string(p.first, p.second));
  201. s = " ";
  202. p = util::stripIter(s.begin(), s.end());
  203. CPPUNIT_ASSERT_EQUAL(str3, std::string(p.first, p.second));
  204. s = " ";
  205. p = util::stripIter(s.begin(), s.end());
  206. CPPUNIT_ASSERT_EQUAL(str3, std::string(p.first, p.second));
  207. std::string str4 = "A";
  208. s = "A";
  209. p = util::stripIter(s.begin(), s.end());
  210. CPPUNIT_ASSERT_EQUAL(str4, std::string(p.first, p.second));
  211. s = " A ";
  212. p = util::stripIter(s.begin(), s.end());
  213. CPPUNIT_ASSERT_EQUAL(str4, std::string(p.first, p.second));
  214. s = " A ";
  215. p = util::stripIter(s.begin(), s.end());
  216. CPPUNIT_ASSERT_EQUAL(str4, std::string(p.first, p.second));
  217. }
  218. void UtilTest::testLstripIter()
  219. {
  220. std::string::iterator r;
  221. std::string s = "foo";
  222. r = util::lstripIter(s.begin(), s.end());
  223. CPPUNIT_ASSERT_EQUAL(std::string("foo"), std::string(r, s.end()));
  224. s = " foo bar ";
  225. r = util::lstripIter(s.begin(), s.end());
  226. CPPUNIT_ASSERT_EQUAL(std::string("foo bar "), std::string(r, s.end()));
  227. s = "f";
  228. r = util::lstripIter(s.begin(), s.end());
  229. CPPUNIT_ASSERT_EQUAL(std::string("f"), std::string(r, s.end()));
  230. s = "foo ";
  231. r = util::lstripIter(s.begin(), s.end());
  232. CPPUNIT_ASSERT_EQUAL(std::string("foo "), std::string(r, s.end()));
  233. }
  234. void UtilTest::testLstripIter_char()
  235. {
  236. std::string::iterator r;
  237. std::string s = "foo";
  238. r = util::lstripIter(s.begin(), s.end(), '$');
  239. CPPUNIT_ASSERT_EQUAL(std::string("foo"), std::string(r, s.end()));
  240. s = "$$foo$bar$$";
  241. r = util::lstripIter(s.begin(), s.end(), '$');
  242. CPPUNIT_ASSERT_EQUAL(std::string("foo$bar$$"), std::string(r, s.end()));
  243. s = "f";
  244. r = util::lstripIter(s.begin(), s.end(), '$');
  245. CPPUNIT_ASSERT_EQUAL(std::string("f"), std::string(r, s.end()));
  246. s = "foo$$";
  247. r = util::lstripIter(s.begin(), s.end(), '$');
  248. CPPUNIT_ASSERT_EQUAL(std::string("foo$$"), std::string(r, s.end()));
  249. }
  250. void UtilTest::testDivide() {
  251. std::string s = "name=value";
  252. auto p1 = util::divide(std::begin(s), std::end(s), '=');
  253. CPPUNIT_ASSERT_EQUAL(std::string("name"),
  254. std::string(p1.first.first, p1.first.second));
  255. CPPUNIT_ASSERT_EQUAL(std::string("value"),
  256. std::string(p1.second.first, p1.second.second));
  257. s = " name = value ";
  258. p1 = util::divide(std::begin(s), std::end(s), '=');
  259. CPPUNIT_ASSERT_EQUAL(std::string("name"),
  260. std::string(p1.first.first, p1.first.second));
  261. CPPUNIT_ASSERT_EQUAL(std::string("value"),
  262. std::string(p1.second.first, p1.second.second));
  263. s = "=value";
  264. p1 = util::divide(std::begin(s), std::end(s), '=');
  265. CPPUNIT_ASSERT_EQUAL(std::string(""),
  266. std::string(p1.first.first, p1.first.second));
  267. CPPUNIT_ASSERT_EQUAL(std::string("value"),
  268. std::string(p1.second.first, p1.second.second));
  269. s = "name=";
  270. p1 = util::divide(std::begin(s), std::end(s), '=');
  271. CPPUNIT_ASSERT_EQUAL(std::string("name"),
  272. std::string(p1.first.first, p1.first.second));
  273. CPPUNIT_ASSERT_EQUAL(std::string(""),
  274. std::string(p1.second.first, p1.second.second));
  275. s = "name";
  276. p1 = util::divide(std::begin(s), std::end(s), '=');
  277. CPPUNIT_ASSERT_EQUAL(std::string("name"),
  278. std::string(p1.first.first, p1.first.second));
  279. CPPUNIT_ASSERT_EQUAL(std::string(""),
  280. std::string(p1.second.first, p1.second.second));
  281. }
  282. void UtilTest::testSplit() {
  283. std::vector<std::string> v;
  284. std::string s = "k1; k2;; k3";
  285. util::split(s.begin(), s.end(), std::back_inserter(v), ';', true);
  286. CPPUNIT_ASSERT_EQUAL((size_t)3, v.size());
  287. std::vector<std::string>::iterator itr = v.begin();
  288. CPPUNIT_ASSERT_EQUAL(std::string("k1"), *itr++);
  289. CPPUNIT_ASSERT_EQUAL(std::string("k2"), *itr++);
  290. CPPUNIT_ASSERT_EQUAL(std::string("k3"), *itr++);
  291. v.clear();
  292. s = "k1; k2; k3";
  293. util::split(s.begin(), s.end(), std::back_inserter(v), ';');
  294. CPPUNIT_ASSERT_EQUAL((size_t)3, v.size());
  295. itr = v.begin();
  296. CPPUNIT_ASSERT_EQUAL(std::string("k1"), *itr++);
  297. CPPUNIT_ASSERT_EQUAL(std::string(" k2"), *itr++);
  298. CPPUNIT_ASSERT_EQUAL(std::string(" k3"), *itr++);
  299. v.clear();
  300. s = "k=v";
  301. util::split(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  302. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  303. itr = v.begin();
  304. CPPUNIT_ASSERT_EQUAL(std::string("k=v"), *itr++);
  305. v.clear();
  306. s = ";;k1;;k2;";
  307. util::split(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  308. CPPUNIT_ASSERT_EQUAL((size_t)6, v.size());
  309. itr = v.begin();
  310. CPPUNIT_ASSERT_EQUAL(std::string(""), *itr++);
  311. CPPUNIT_ASSERT_EQUAL(std::string(""), *itr++);
  312. CPPUNIT_ASSERT_EQUAL(std::string("k1"), *itr++);
  313. CPPUNIT_ASSERT_EQUAL(std::string(""), *itr++);
  314. CPPUNIT_ASSERT_EQUAL(std::string("k2"), *itr++);
  315. CPPUNIT_ASSERT_EQUAL(std::string(""), *itr++);
  316. v.clear();
  317. s = ";;k1;;k2;";
  318. util::split(s.begin(), s.end(), std::back_inserter(v), ';');
  319. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  320. itr = v.begin();
  321. CPPUNIT_ASSERT_EQUAL(std::string("k1"), *itr++);
  322. CPPUNIT_ASSERT_EQUAL(std::string("k2"), *itr++);
  323. v.clear();
  324. s = "k; ";
  325. util::split(s.begin(), s.end(), std::back_inserter(v), ';');
  326. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  327. itr = v.begin();
  328. CPPUNIT_ASSERT_EQUAL(std::string("k"), *itr++);
  329. CPPUNIT_ASSERT_EQUAL(std::string(" "), *itr++);
  330. v.clear();
  331. s = " ";
  332. util::split(s.begin(), s.end(), std::back_inserter(v), ';', true, true);
  333. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  334. CPPUNIT_ASSERT_EQUAL(std::string(""), v[0]);
  335. v.clear();
  336. s = " ";
  337. util::split(s.begin(), s.end(), std::back_inserter(v), ';', true);
  338. CPPUNIT_ASSERT_EQUAL((size_t)0, v.size());
  339. v.clear();
  340. s = " ";
  341. util::split(s.begin(), s.end(), std::back_inserter(v), ';');
  342. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  343. CPPUNIT_ASSERT_EQUAL(std::string(" "), v[0]);
  344. v.clear();
  345. s = ";";
  346. util::split(s.begin(), s.end(), std::back_inserter(v), ';');
  347. CPPUNIT_ASSERT_EQUAL((size_t)0, v.size());
  348. v.clear();
  349. s = ";";
  350. util::split(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  351. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  352. itr = v.begin();
  353. CPPUNIT_ASSERT_EQUAL(std::string(""), *itr++);
  354. CPPUNIT_ASSERT_EQUAL(std::string(""), *itr++);
  355. v.clear();
  356. s = "";
  357. util::split(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  358. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  359. CPPUNIT_ASSERT_EQUAL(std::string(""), v[0]);
  360. }
  361. void UtilTest::testSplitIter() {
  362. std::vector<Scip> v;
  363. std::string s = "k1; k2;; k3";
  364. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';', true);
  365. CPPUNIT_ASSERT_EQUAL((size_t)3, v.size());
  366. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  367. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[1].first, v[1].second));
  368. CPPUNIT_ASSERT_EQUAL(std::string("k3"), std::string(v[2].first, v[2].second));
  369. v.clear();
  370. s = "k1; k2; k3";
  371. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';');
  372. CPPUNIT_ASSERT_EQUAL((size_t)3, v.size());
  373. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  374. CPPUNIT_ASSERT_EQUAL(std::string(" k2"),
  375. std::string(v[1].first, v[1].second));
  376. CPPUNIT_ASSERT_EQUAL(std::string(" k3"),
  377. std::string(v[2].first, v[2].second));
  378. v.clear();
  379. s = "k=v";
  380. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  381. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  382. CPPUNIT_ASSERT_EQUAL(std::string("k=v"),
  383. std::string(v[0].first, v[0].second));
  384. v.clear();
  385. s = ";;k1;;k2;";
  386. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  387. CPPUNIT_ASSERT_EQUAL((size_t)6, v.size());
  388. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  389. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[1].first, v[1].second));
  390. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[2].first, v[2].second));
  391. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[3].first, v[3].second));
  392. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[4].first, v[4].second));
  393. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[5].first, v[5].second));
  394. v.clear();
  395. s = ";;k1;;k2;";
  396. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';');
  397. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  398. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  399. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[1].first, v[1].second));
  400. v.clear();
  401. s = "k; ";
  402. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';');
  403. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  404. CPPUNIT_ASSERT_EQUAL(std::string("k"), std::string(v[0].first, v[0].second));
  405. CPPUNIT_ASSERT_EQUAL(std::string(" "), std::string(v[1].first, v[1].second));
  406. v.clear();
  407. s = " ";
  408. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';', true, true);
  409. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  410. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  411. v.clear();
  412. s = " ";
  413. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';', true);
  414. CPPUNIT_ASSERT_EQUAL((size_t)0, v.size());
  415. v.clear();
  416. s = " ";
  417. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';');
  418. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  419. CPPUNIT_ASSERT_EQUAL(std::string(" "), std::string(v[0].first, v[0].second));
  420. v.clear();
  421. s = ";";
  422. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';');
  423. CPPUNIT_ASSERT_EQUAL((size_t)0, v.size());
  424. v.clear();
  425. s = ";";
  426. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  427. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  428. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  429. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[1].first, v[1].second));
  430. v.clear();
  431. s = "";
  432. util::splitIter(s.begin(), s.end(), std::back_inserter(v), ';', false, true);
  433. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  434. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  435. }
  436. void UtilTest::testSplitIterM() {
  437. const char d[] = ";";
  438. const char md[] = "; ";
  439. std::vector<Scip> v;
  440. std::string s = "k1; k2;; k3";
  441. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d, true);
  442. CPPUNIT_ASSERT_EQUAL((size_t)3, v.size());
  443. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  444. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[1].first, v[1].second));
  445. CPPUNIT_ASSERT_EQUAL(std::string("k3"), std::string(v[2].first, v[2].second));
  446. v.clear();
  447. s = "k1; k2; k3";
  448. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d);
  449. CPPUNIT_ASSERT_EQUAL((size_t)3, v.size());
  450. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  451. CPPUNIT_ASSERT_EQUAL(std::string(" k2"),
  452. std::string(v[1].first, v[1].second));
  453. CPPUNIT_ASSERT_EQUAL(std::string(" k3"),
  454. std::string(v[2].first, v[2].second));
  455. v.clear();
  456. s = "k1; k2; k3";
  457. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), md);
  458. CPPUNIT_ASSERT_EQUAL((size_t)3, v.size());
  459. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  460. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[1].first, v[1].second));
  461. CPPUNIT_ASSERT_EQUAL(std::string("k3"), std::string(v[2].first, v[2].second));
  462. v.clear();
  463. s = "k1; k2; k3;";
  464. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), md, false, true);
  465. CPPUNIT_ASSERT_EQUAL((size_t)6, v.size());
  466. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  467. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[1].first, v[1].second));
  468. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[2].first, v[2].second));
  469. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[3].first, v[3].second));
  470. CPPUNIT_ASSERT_EQUAL(std::string("k3"), std::string(v[4].first, v[4].second));
  471. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[5].first, v[5].second));
  472. v.clear();
  473. s = "k=v";
  474. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d, false, true);
  475. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  476. CPPUNIT_ASSERT_EQUAL(std::string("k=v"),
  477. std::string(v[0].first, v[0].second));
  478. v.clear();
  479. s = ";;k1;;k2;";
  480. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d, false, true);
  481. CPPUNIT_ASSERT_EQUAL((size_t)6, v.size());
  482. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  483. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[1].first, v[1].second));
  484. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[2].first, v[2].second));
  485. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[3].first, v[3].second));
  486. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[4].first, v[4].second));
  487. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[5].first, v[5].second));
  488. v.clear();
  489. s = ";;k1;;k2;";
  490. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d);
  491. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  492. CPPUNIT_ASSERT_EQUAL(std::string("k1"), std::string(v[0].first, v[0].second));
  493. CPPUNIT_ASSERT_EQUAL(std::string("k2"), std::string(v[1].first, v[1].second));
  494. v.clear();
  495. s = "k; ";
  496. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d);
  497. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  498. CPPUNIT_ASSERT_EQUAL(std::string("k"), std::string(v[0].first, v[0].second));
  499. CPPUNIT_ASSERT_EQUAL(std::string(" "), std::string(v[1].first, v[1].second));
  500. v.clear();
  501. s = " ";
  502. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d, true, true);
  503. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  504. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  505. v.clear();
  506. s = " ";
  507. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d, true);
  508. CPPUNIT_ASSERT_EQUAL((size_t)0, v.size());
  509. v.clear();
  510. s = " ";
  511. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d);
  512. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  513. CPPUNIT_ASSERT_EQUAL(std::string(" "), std::string(v[0].first, v[0].second));
  514. v.clear();
  515. s = ";";
  516. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d);
  517. CPPUNIT_ASSERT_EQUAL((size_t)0, v.size());
  518. v.clear();
  519. s = ";";
  520. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d, false, true);
  521. CPPUNIT_ASSERT_EQUAL((size_t)2, v.size());
  522. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  523. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[1].first, v[1].second));
  524. v.clear();
  525. s = "";
  526. util::splitIterM(s.begin(), s.end(), std::back_inserter(v), d, false, true);
  527. CPPUNIT_ASSERT_EQUAL((size_t)1, v.size());
  528. CPPUNIT_ASSERT_EQUAL(std::string(""), std::string(v[0].first, v[0].second));
  529. }
  530. void UtilTest::testEndsWith() {
  531. std::string target = "abcdefg";
  532. std::string part = "fg";
  533. CPPUNIT_ASSERT(util::endsWith(target.begin(), target.end(),
  534. part.begin(), part.end()));
  535. target = "abdefg";
  536. part = "g";
  537. CPPUNIT_ASSERT(util::endsWith(target.begin(), target.end(),
  538. part.begin(), part.end()));
  539. target = "abdefg";
  540. part = "eg";
  541. CPPUNIT_ASSERT(!util::endsWith(target.begin(), target.end(),
  542. part.begin(), part.end()));
  543. target = "g";
  544. part = "eg";
  545. CPPUNIT_ASSERT(!util::endsWith(target.begin(), target.end(),
  546. part.begin(), part.end()));
  547. target = "g";
  548. part = "g";
  549. CPPUNIT_ASSERT(util::endsWith(target.begin(), target.end(),
  550. part.begin(), part.end()));
  551. target = "g";
  552. part = "";
  553. CPPUNIT_ASSERT(util::endsWith(target.begin(), target.end(),
  554. part.begin(), part.end()));
  555. target = "";
  556. part = "";
  557. CPPUNIT_ASSERT(util::endsWith(target.begin(), target.end(),
  558. part.begin(), part.end()));
  559. target = "";
  560. part = "g";
  561. CPPUNIT_ASSERT(!util::endsWith(target.begin(), target.end(),
  562. part.begin(), part.end()));
  563. }
  564. void UtilTest::testIendsWith() {
  565. std::string target = "abcdefg";
  566. std::string part = "Fg";
  567. CPPUNIT_ASSERT(util::iendsWith(target.begin(), target.end(),
  568. part.begin(), part.end()));
  569. target = "abdefg";
  570. part = "ef";
  571. CPPUNIT_ASSERT(!util::iendsWith(target.begin(), target.end(),
  572. part.begin(), part.end()));
  573. }
  574. void UtilTest::testStreq()
  575. {
  576. std::string s1, s2;
  577. s1 = "foo";
  578. s2 = "foo";
  579. CPPUNIT_ASSERT(util::streq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  580. CPPUNIT_ASSERT(util::streq(s1.begin(), s1.end(), s2.c_str()));
  581. s2 = "fooo";
  582. CPPUNIT_ASSERT(!util::streq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  583. CPPUNIT_ASSERT(!util::streq(s1.begin(), s1.end(), s2.c_str()));
  584. s2 = "fo";
  585. CPPUNIT_ASSERT(!util::streq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  586. CPPUNIT_ASSERT(!util::streq(s1.begin(), s1.end(), s2.c_str()));
  587. s2 = "";
  588. CPPUNIT_ASSERT(!util::streq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  589. CPPUNIT_ASSERT(!util::streq(s1.begin(), s1.end(), s2.c_str()));
  590. s1 = "";
  591. CPPUNIT_ASSERT(util::streq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  592. CPPUNIT_ASSERT(util::streq(s1.begin(), s1.end(), s2.c_str()));
  593. }
  594. void UtilTest::testStrieq()
  595. {
  596. std::string s1, s2;
  597. s1 = "foo";
  598. s2 = "foo";
  599. CPPUNIT_ASSERT(util::strieq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  600. CPPUNIT_ASSERT(util::strieq(s1.begin(), s1.end(), s2.c_str()));
  601. s1 = "FoO";
  602. s2 = "fOo";
  603. CPPUNIT_ASSERT(util::strieq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  604. CPPUNIT_ASSERT(util::strieq(s1.begin(), s1.end(), s2.c_str()));
  605. s2 = "fooo";
  606. CPPUNIT_ASSERT(!util::strieq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  607. CPPUNIT_ASSERT(!util::strieq(s1.begin(), s1.end(), s2.c_str()));
  608. s2 = "fo";
  609. CPPUNIT_ASSERT(!util::strieq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  610. CPPUNIT_ASSERT(!util::strieq(s1.begin(), s1.end(), s2.c_str()));
  611. s2 = "";
  612. CPPUNIT_ASSERT(!util::strieq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  613. CPPUNIT_ASSERT(!util::strieq(s1.begin(), s1.end(), s2.c_str()));
  614. s1 = "";
  615. CPPUNIT_ASSERT(util::strieq(s1.begin(), s1.end(), s2.begin(), s2.end()));
  616. CPPUNIT_ASSERT(util::strieq(s1.begin(), s1.end(), s2.c_str()));
  617. }
  618. void UtilTest::testStrifind()
  619. {
  620. std::string s1, s2;
  621. s1 = "yamagakani mukashi wo toheba hARU no tuki";
  622. s2 = "HaRu";
  623. CPPUNIT_ASSERT(util::strifind(s1.begin(), s1.end(), s2.begin(), s2.end())
  624. != s1.end());
  625. s2 = "aki";
  626. CPPUNIT_ASSERT(util::strifind(s1.begin(), s1.end(), s2.begin(), s2.end())
  627. == s1.end());
  628. s1 = "h";
  629. s2 = "HH";
  630. CPPUNIT_ASSERT(util::strifind(s1.begin(), s1.end(), s2.begin(), s2.end())
  631. == s1.end());
  632. }
  633. void UtilTest::testReplace() {
  634. CPPUNIT_ASSERT_EQUAL(std::string("abc\n"), util::replace("abc\r\n", "\r", ""));
  635. CPPUNIT_ASSERT_EQUAL(std::string("abc"), util::replace("abc\r\n", "\r\n", ""));
  636. CPPUNIT_ASSERT_EQUAL(std::string(""), util::replace("", "\r\n", ""));
  637. CPPUNIT_ASSERT_EQUAL(std::string("abc"), util::replace("abc", "", "a"));
  638. CPPUNIT_ASSERT_EQUAL(std::string("xbc"), util::replace("abc", "a", "x"));
  639. }
  640. void UtilTest::testStartsWith() {
  641. std::string target;
  642. std::string part;
  643. target = "abcdefg";
  644. part = "abc";
  645. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(),
  646. part.begin(), part.end()));
  647. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(), part.c_str()));
  648. target = "abcdefg";
  649. part = "abx";
  650. CPPUNIT_ASSERT(!util::startsWith(target.begin(), target.end(),
  651. part.begin(), part.end()));
  652. CPPUNIT_ASSERT(!util::startsWith(target.begin(), target.end(), part.c_str()));
  653. target = "abcdefg";
  654. part = "bcd";
  655. CPPUNIT_ASSERT(!util::startsWith(target.begin(), target.end(),
  656. part.begin(), part.end()));
  657. CPPUNIT_ASSERT(!util::startsWith(target.begin(), target.end(), part.c_str()));
  658. target = "";
  659. part = "a";
  660. CPPUNIT_ASSERT(!util::startsWith(target.begin(), target.end(),
  661. part.begin(), part.end()));
  662. CPPUNIT_ASSERT(!util::startsWith(target.begin(), target.end(), part.c_str()));
  663. target = "";
  664. part = "";
  665. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(),
  666. part.begin(), part.end()));
  667. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(), part.c_str()));
  668. target = "a";
  669. part = "";
  670. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(),
  671. part.begin(), part.end()));
  672. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(), part.c_str()));
  673. target = "a";
  674. part = "a";
  675. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(),
  676. part.begin(), part.end()));
  677. CPPUNIT_ASSERT(util::startsWith(target.begin(), target.end(), part.c_str()));
  678. }
  679. void UtilTest::testIstartsWith() {
  680. std::string target;
  681. std::string part;
  682. target = "abcdefg";
  683. part = "aBc";
  684. CPPUNIT_ASSERT(util::istartsWith(target.begin(), target.end(),
  685. part.begin(), part.end()));
  686. CPPUNIT_ASSERT(util::istartsWith(target.begin(), target.end(), part.c_str()));
  687. target = "abcdefg";
  688. part = "abx";
  689. CPPUNIT_ASSERT(!util::istartsWith(target.begin(), target.end(),
  690. part.begin(), part.end()));
  691. CPPUNIT_ASSERT(!util::istartsWith(target.begin(), target.end(), part.c_str()));
  692. }
  693. void UtilTest::testGetContentDispositionFilename() {
  694. std::string val;
  695. val = "attachment; filename=\"aria2.tar.bz2\"";
  696. CPPUNIT_ASSERT_EQUAL(std::string("aria2.tar.bz2"),
  697. util::getContentDispositionFilename(val));
  698. val = "attachment; filename=\"\"";
  699. CPPUNIT_ASSERT_EQUAL(std::string(""),
  700. util::getContentDispositionFilename(val));
  701. val = "attachment; filename=\"";
  702. CPPUNIT_ASSERT_EQUAL(std::string(""),
  703. util::getContentDispositionFilename(val));
  704. val = "attachment; filename= \" aria2.tar.bz2 \"";
  705. CPPUNIT_ASSERT_EQUAL(std::string(" aria2.tar.bz2 "),
  706. util::getContentDispositionFilename(val));
  707. val = "attachment; filename=dir/file";
  708. CPPUNIT_ASSERT_EQUAL(std::string(""),
  709. util::getContentDispositionFilename(val));
  710. val = "attachment; filename=dir\\file";
  711. CPPUNIT_ASSERT_EQUAL(std::string(""),
  712. util::getContentDispositionFilename(val));
  713. val = "attachment; filename=\"dir/file\"";
  714. CPPUNIT_ASSERT_EQUAL(std::string(""),
  715. util::getContentDispositionFilename(val));
  716. val = "attachment; filename=\"dir\\\\file\"";
  717. CPPUNIT_ASSERT_EQUAL(std::string(""),
  718. util::getContentDispositionFilename(val));
  719. val = "attachment; filename=\"/etc/passwd\"";
  720. CPPUNIT_ASSERT_EQUAL(std::string(""),
  721. util::getContentDispositionFilename(val));
  722. val = "attachment; filename=\"..\"";
  723. CPPUNIT_ASSERT_EQUAL(std::string(""),
  724. util::getContentDispositionFilename(val));
  725. val = "attachment; filename=..";
  726. CPPUNIT_ASSERT_EQUAL(std::string(""),
  727. util::getContentDispositionFilename(val));
  728. // Unescaping %2E%2E%2F produces "../". But since we won't unescape,
  729. // we just accept it as is.
  730. val = "attachment; filename=\"%2E%2E%2Ffoo.html\"";
  731. CPPUNIT_ASSERT_EQUAL(std::string("%2E%2E%2Ffoo.html"),
  732. util::getContentDispositionFilename(val));
  733. // iso-8859-1 string will be converted to utf-8.
  734. val = "attachment; filename*=iso-8859-1''foo-%E4.html";
  735. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  736. util::getContentDispositionFilename(val));
  737. val = "attachment; filename*= UTF-8''foo-%c3%a4.html";
  738. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  739. util::getContentDispositionFilename(val));
  740. // iso-8859-1 string will be converted to utf-8.
  741. val = "attachment; filename=\"foo-%E4.html\"";
  742. val = util::percentDecode(val.begin(), val.end());
  743. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  744. util::getContentDispositionFilename(val));
  745. }
  746. void UtilTest::testParseContentDisposition() {
  747. char dest[1024];
  748. size_t destlen = sizeof(dest);
  749. const char *cs;
  750. size_t cslen;
  751. std::string val;
  752. // test cases from http://greenbytes.de/tech/tc2231/
  753. // inlonly
  754. val = "inline";
  755. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  756. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  757. // inlonlyquoted
  758. val = "\"inline\"";
  759. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  760. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  761. // inlwithasciifilename
  762. val = "inline; filename=\"foo.html\"";
  763. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  764. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  765. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  766. std::string(&dest[0], &dest[8]));
  767. // inlwithfnattach
  768. val = "inline; filename=\"Not an attachment!\"";
  769. CPPUNIT_ASSERT_EQUAL((ssize_t)18, util::parse_content_disposition
  770. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  771. CPPUNIT_ASSERT_EQUAL(std::string("Not an attachment!"),
  772. std::string(&dest[0], &dest[18]));
  773. // inlwithasciifilenamepdf
  774. val = "inline; filename=\"foo.pdf\"";
  775. CPPUNIT_ASSERT_EQUAL((ssize_t)7, util::parse_content_disposition
  776. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  777. CPPUNIT_ASSERT_EQUAL(std::string("foo.pdf"),
  778. std::string(&dest[0], &dest[7]));
  779. // attwithasciifilename25
  780. val = "attachment; filename=\"0000000000111111111122222\"";
  781. CPPUNIT_ASSERT_EQUAL((ssize_t)25, util::parse_content_disposition
  782. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  783. CPPUNIT_ASSERT_EQUAL(std::string("0000000000111111111122222"),
  784. std::string(&dest[0], &dest[25]));
  785. // attwithasciifilename35
  786. val = "attachment; filename=\"00000000001111111111222222222233333\"";
  787. CPPUNIT_ASSERT_EQUAL((ssize_t)35, util::parse_content_disposition
  788. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  789. CPPUNIT_ASSERT_EQUAL(std::string("00000000001111111111222222222233333"),
  790. std::string(&dest[0], &dest[35]));
  791. // attwithasciifnescapedchar
  792. val = "attachment; filename=\"f\\oo.html\"";
  793. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  794. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  795. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  796. std::string(&dest[0], &dest[8]));
  797. // attwithasciifnescapedquote
  798. val = "attachment; filename=\"\\\"quoting\\\" tested.html\"";
  799. CPPUNIT_ASSERT_EQUAL((ssize_t)21, util::parse_content_disposition
  800. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  801. CPPUNIT_ASSERT_EQUAL(std::string("\"quoting\" tested.html"),
  802. std::string(&dest[0], &dest[21]));
  803. // attwithquotedsemicolon
  804. val = "attachment; filename=\"Here's a semicolon;.html\"";
  805. CPPUNIT_ASSERT_EQUAL((ssize_t)24, util::parse_content_disposition
  806. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  807. CPPUNIT_ASSERT_EQUAL(std::string("Here's a semicolon;.html"),
  808. std::string(&dest[0], &dest[24]));
  809. // attwithfilenameandextparam
  810. val = "attachment; foo=\"bar\"; filename=\"foo.html\"";
  811. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  812. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  813. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  814. std::string(&dest[0], &dest[8]));
  815. // attwithfilenameandextparamescaped
  816. val = "attachment; foo=\"\\\"\\\\\";filename=\"foo.html\"";
  817. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  818. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  819. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  820. std::string(&dest[0], &dest[8]));
  821. // attwithasciifilenameucase
  822. val = "attachment; FILENAME=\"foo.html\"";
  823. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  824. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  825. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  826. std::string(&dest[0], &dest[8]));
  827. // attwithasciifilenamenq
  828. val = "attachment; filename=foo.html";
  829. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  830. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  831. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  832. std::string(&dest[0], &dest[8]));
  833. // attwithtokfncommanq
  834. val = "attachment; filename=foo,bar.html";
  835. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  836. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  837. // attwithasciifilenamenqs
  838. val = "attachment; filename=foo.html ;";
  839. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  840. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  841. // attemptyparam
  842. val = "attachment; ;filename=foo";
  843. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  844. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  845. // attwithasciifilenamenqws
  846. val = "attachment; filename=foo bar.html";
  847. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  848. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  849. // attwithfntokensq
  850. val = "attachment; filename='foo.bar'";
  851. CPPUNIT_ASSERT_EQUAL((ssize_t)9, util::parse_content_disposition
  852. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  853. CPPUNIT_ASSERT_EQUAL(std::string("'foo.bar'"),
  854. std::string(&dest[0], &dest[9]));
  855. // attwithisofnplain
  856. // attachment; filename="foo-ä.html"
  857. val = "attachment; filename=\"foo-%E4.html\"";
  858. val = util::percentDecode(val.begin(), val.end());
  859. CPPUNIT_ASSERT_EQUAL((ssize_t)10, util::parse_content_disposition
  860. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  861. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  862. util::iso8859p1ToUtf8(std::string(&dest[0], &dest[10])));
  863. // attwithutf8fnplain
  864. // attachment; filename="foo-ä.html"
  865. val = "attachment; filename=\"foo-%C3%A4.html\"";
  866. val = util::percentDecode(val.begin(), val.end());
  867. CPPUNIT_ASSERT_EQUAL((ssize_t)11, util::parse_content_disposition
  868. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  869. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  870. util::iso8859p1ToUtf8(std::string(&dest[0], &dest[11])));
  871. // attwithfnrawpctenca
  872. val = "attachment; filename=\"foo-%41.html\"";
  873. CPPUNIT_ASSERT_EQUAL((ssize_t)12, util::parse_content_disposition
  874. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  875. CPPUNIT_ASSERT_EQUAL(std::string("foo-%41.html"),
  876. std::string(&dest[0], &dest[12]));
  877. // attwithfnusingpct
  878. val = "attachment; filename=\"50%.html\"";
  879. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  880. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  881. CPPUNIT_ASSERT_EQUAL(std::string("50%.html"),
  882. std::string(&dest[0], &dest[8]));
  883. // attwithfnrawpctencaq
  884. val = "attachment; filename=\"foo-%\\41.html\"";
  885. CPPUNIT_ASSERT_EQUAL((ssize_t)12, util::parse_content_disposition
  886. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  887. CPPUNIT_ASSERT_EQUAL(std::string("foo-%41.html"),
  888. std::string(&dest[0], &dest[12]));
  889. // attwithnamepct
  890. val = "attachment; name=\"foo-%41.html\"";
  891. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  892. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  893. // attwithfilenamepctandiso
  894. // attachment; filename="ä-%41.html"
  895. val = "attachment; filename=\"%E4-%2541.html\"";
  896. val = util::percentDecode(val.begin(), val.end());
  897. CPPUNIT_ASSERT_EQUAL((ssize_t)10, util::parse_content_disposition
  898. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  899. CPPUNIT_ASSERT_EQUAL(std::string("ä-%41.html"),
  900. util::iso8859p1ToUtf8(std::string(&dest[0], &dest[10])));
  901. // attwithfnrawpctenclong
  902. val = "attachment; filename=\"foo-%c3%a4-%e2%82%ac.html\"";
  903. CPPUNIT_ASSERT_EQUAL((ssize_t)25, util::parse_content_disposition
  904. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  905. CPPUNIT_ASSERT_EQUAL(std::string("foo-%c3%a4-%e2%82%ac.html"),
  906. std::string(&dest[0], &dest[25]));
  907. // attwithasciifilenamews1
  908. val = "attachment; filename =\"foo.html\"";
  909. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  910. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  911. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  912. std::string(&dest[0], &dest[8]));
  913. // attwith2filenames
  914. val = "attachment; filename=\"foo.html\"; filename=\"bar.html\"";
  915. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  916. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  917. // attfnbrokentoken
  918. val = "attachment; filename=foo[1](2).html";
  919. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  920. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  921. // attfnbrokentokeniso
  922. val = "attachment; filename=foo-%E4.html";
  923. val = util::percentDecode(val.begin(), val.end());
  924. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  925. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  926. // attfnbrokentokenutf
  927. // attachment; filename=foo-ä.html
  928. val = "attachment; filename=foo-ä.html";
  929. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  930. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  931. // attmissingdisposition
  932. val = "filename=foo.html";
  933. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  934. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  935. // attmissingdisposition2
  936. val = "x=y; filename=foo.html";
  937. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  938. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  939. // attmissingdisposition3
  940. val = "\"foo; filename=bar;baz\"; filename=qux";
  941. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  942. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  943. // attmissingdisposition4
  944. val = "filename=foo.html, filename=bar.html";
  945. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  946. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  947. // emptydisposition
  948. val = "; filename=foo.html";
  949. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  950. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  951. // doublecolon
  952. val = ": inline; attachment; filename=foo.html";
  953. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  954. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  955. // attandinline
  956. val = "inline; attachment; filename=foo.html";
  957. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  958. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  959. // attandinline2
  960. val = "attachment; inline; filename=foo.html";
  961. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  962. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  963. // attbrokenquotedfn
  964. val = "attachment; filename=\"foo.html\".txt";
  965. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  966. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  967. // attbrokenquotedfn2
  968. val = "attachment; filename=\"bar";
  969. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  970. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  971. // attbrokenquotedfn3
  972. val = "attachment; filename=foo\"bar;baz\"qux";
  973. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  974. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  975. // attmultinstances
  976. val = "attachment; filename=foo.html, attachment; filename=bar.html";
  977. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  978. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  979. // attmissingdelim
  980. val = "attachment; foo=foo filename=bar";
  981. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  982. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  983. // attmissingdelim2
  984. val = "attachment; filename=bar foo=foo ";
  985. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  986. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  987. // attmissingdelim3
  988. val = "attachment filename=bar";
  989. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  990. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  991. // attreversed
  992. val = "filename=foo.html; attachment";
  993. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  994. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  995. // attconfusedparam
  996. val = "attachment; xfilename=foo.html";
  997. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  998. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  999. // attabspath
  1000. val = "attachment; filename=\"/foo.html\"";
  1001. CPPUNIT_ASSERT_EQUAL((ssize_t)9, util::parse_content_disposition
  1002. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1003. CPPUNIT_ASSERT_EQUAL(std::string("/foo.html"),
  1004. std::string(&dest[0], &dest[9]));
  1005. // attabspathwin
  1006. val = "attachment; filename=\"\\\\foo.html\"";
  1007. CPPUNIT_ASSERT_EQUAL((ssize_t)9, util::parse_content_disposition
  1008. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1009. CPPUNIT_ASSERT_EQUAL(std::string("\\foo.html"),
  1010. std::string(&dest[0], &dest[9]));
  1011. // attcdate
  1012. val = "attachment; creation-date=\"Wed, 12 Feb 1997 16:29:51 -0500\"";
  1013. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  1014. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1015. // dispext
  1016. val = "foobar";
  1017. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  1018. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1019. // dispextbadfn
  1020. val = "attachment; example=\"filename=example.txt\"";
  1021. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  1022. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1023. // attwithisofn2231iso
  1024. val = "attachment; filename*=iso-8859-1''foo-%E4.html";
  1025. CPPUNIT_ASSERT_EQUAL((ssize_t)10, util::parse_content_disposition
  1026. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1027. CPPUNIT_ASSERT_EQUAL(std::string("iso-8859-1"), std::string(cs, cslen));
  1028. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  1029. util::iso8859p1ToUtf8(std::string(&dest[0], &dest[10])));
  1030. // attwithfn2231utf8
  1031. val = "attachment; filename*=UTF-8''foo-%c3%a4-%e2%82%ac.html";
  1032. CPPUNIT_ASSERT_EQUAL((ssize_t)15, util::parse_content_disposition
  1033. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1034. CPPUNIT_ASSERT_EQUAL(std::string("UTF-8"), std::string(cs, cslen));
  1035. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä-€.html"),
  1036. std::string(&dest[0], &dest[15]));
  1037. // attwithfn2231noc
  1038. val = "attachment; filename*=''foo-%c3%a4-%e2%82%ac.html";
  1039. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1040. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1041. // attwithfn2231utf8comp
  1042. val = "attachment; filename*=UTF-8''foo-a%cc%88.html";
  1043. CPPUNIT_ASSERT_EQUAL((ssize_t)12, util::parse_content_disposition
  1044. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1045. val = "foo-a%cc%88.html";
  1046. CPPUNIT_ASSERT_EQUAL(std::string(util::percentDecode(val.begin(),
  1047. val.end())),
  1048. std::string(&dest[0], &dest[12]));
  1049. // attwithfn2231utf8-bad
  1050. val = "attachment; filename*=iso-8859-1''foo-%c3%a4-%e2%82%ac.html";
  1051. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1052. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1053. // attwithfn2231iso-bad
  1054. val = "attachment; filename*=utf-8''foo-%E4.html";
  1055. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1056. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1057. // attwithfn2231ws1
  1058. val = "attachment; filename *=UTF-8''foo-%c3%a4.html";
  1059. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1060. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1061. // attwithfn2231ws2
  1062. val = "attachment; filename*= UTF-8''foo-%c3%a4.html";
  1063. CPPUNIT_ASSERT_EQUAL((ssize_t)11, util::parse_content_disposition
  1064. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1065. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  1066. std::string(&dest[0], &dest[11]));
  1067. // attwithfn2231ws3
  1068. val = "attachment; filename* =UTF-8''foo-%c3%a4.html";
  1069. CPPUNIT_ASSERT_EQUAL((ssize_t)11, util::parse_content_disposition
  1070. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1071. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  1072. std::string(&dest[0], &dest[11]));
  1073. // attwithfn2231quot
  1074. val = "attachment; filename*=\"UTF-8''foo-%c3%a4.html\"";
  1075. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1076. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1077. // attwithfn2231quot2
  1078. val = "attachment; filename*=\"foo%20bar.html\"";
  1079. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1080. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1081. // attwithfn2231singleqmissing
  1082. val = "attachment; filename*=UTF-8'foo-%c3%a4.html";
  1083. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1084. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1085. // attwithfn2231nbadpct1
  1086. val = "attachment; filename*=UTF-8''foo%";
  1087. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1088. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1089. // attwithfn2231nbadpct2
  1090. val = "attachment; filename*=UTF-8''f%oo.html";
  1091. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1092. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1093. // attwithfn2231dpct
  1094. val = "attachment; filename*=UTF-8''A-%2541.html";
  1095. CPPUNIT_ASSERT_EQUAL((ssize_t)10, util::parse_content_disposition
  1096. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1097. CPPUNIT_ASSERT_EQUAL(std::string("A-%41.html"),
  1098. std::string(&dest[0], &dest[10]));
  1099. // attwithfn2231abspathdisguised
  1100. val = "attachment; filename*=UTF-8''%5cfoo.html";
  1101. CPPUNIT_ASSERT_EQUAL((ssize_t)9, util::parse_content_disposition
  1102. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1103. CPPUNIT_ASSERT_EQUAL(std::string("\\foo.html"),
  1104. std::string(&dest[0], &dest[9]));
  1105. // attfnboth
  1106. val = "attachment; filename=\"foo-ae.html\"; filename*=UTF-8''foo-%c3%a4.html";
  1107. CPPUNIT_ASSERT_EQUAL((ssize_t)11, util::parse_content_disposition
  1108. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1109. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  1110. std::string(&dest[0], &dest[11]));
  1111. // attfnboth2
  1112. val = "attachment; filename*=UTF-8''foo-%c3%a4.html; filename=\"foo-ae.html\"";
  1113. CPPUNIT_ASSERT_EQUAL((ssize_t)11, util::parse_content_disposition
  1114. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1115. CPPUNIT_ASSERT_EQUAL(std::string("foo-ä.html"),
  1116. std::string(&dest[0], &dest[11]));
  1117. // attfnboth3
  1118. val = "attachment; filename*0*=ISO-8859-15''euro-sign%3d%a4; filename*=ISO-8859-1''currency-sign%3d%a4";
  1119. CPPUNIT_ASSERT_EQUAL((ssize_t)15, util::parse_content_disposition
  1120. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1121. CPPUNIT_ASSERT_EQUAL(std::string("ISO-8859-1"), std::string(cs, cslen));
  1122. CPPUNIT_ASSERT_EQUAL(std::string("currency-sign=¤"),
  1123. util::iso8859p1ToUtf8(std::string(&dest[0], &dest[15])));
  1124. // attnewandfn
  1125. val = "attachment; foobar=x; filename=\"foo.html\"";
  1126. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1127. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1128. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1129. std::string(&dest[0], &dest[8]));
  1130. // attrfc2047token
  1131. val = "attachment; filename==?ISO-8859-1?Q?foo-=E4.html?=";
  1132. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1133. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1134. // attrfc2047quoted
  1135. val = "attachment; filename=\"=?ISO-8859-1?Q?foo-=E4.html?=\"";
  1136. CPPUNIT_ASSERT_EQUAL((ssize_t)29, util::parse_content_disposition
  1137. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1138. CPPUNIT_ASSERT_EQUAL(std::string("=?ISO-8859-1?Q?foo-=E4.html?="),
  1139. std::string(&dest[0], &dest[29]));
  1140. // aria2 original testcases
  1141. // zero-length filename. token cannot be empty, so this is invalid.
  1142. val = "attachment; filename=";
  1143. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1144. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1145. // zero-length filename. quoted-string can be empty string, so this
  1146. // is ok.
  1147. val = "attachment; filename=\"\"";
  1148. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  1149. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1150. // empty value is not allowed
  1151. val = "attachment; filename=;";
  1152. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1153. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1154. // / is not valid char in token.
  1155. val = "attachment; filename=dir/file";
  1156. CPPUNIT_ASSERT_EQUAL((ssize_t)-1, util::parse_content_disposition
  1157. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1158. // value-chars is *(pct-encoded / attr-char), so empty string is
  1159. // allowed.
  1160. val = "attachment; filename*=UTF-8''";
  1161. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  1162. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1163. CPPUNIT_ASSERT_EQUAL(std::string("UTF-8"), std::string(cs, cslen));
  1164. val = "attachment; filename*=UTF-8''; filename=foo";
  1165. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  1166. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1167. CPPUNIT_ASSERT_EQUAL(std::string("UTF-8"), std::string(cs, cslen));
  1168. val = "attachment; filename*=UTF-8'' ; filename=foo";
  1169. CPPUNIT_ASSERT_EQUAL((ssize_t)0, util::parse_content_disposition
  1170. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1171. CPPUNIT_ASSERT_EQUAL(std::string("UTF-8"), std::string(cs, cslen));
  1172. // with language
  1173. val = "attachment; filename*=UTF-8'japanese'konnichiwa";
  1174. CPPUNIT_ASSERT_EQUAL((ssize_t)10, util::parse_content_disposition
  1175. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1176. CPPUNIT_ASSERT_EQUAL(std::string("konnichiwa"),
  1177. std::string(&dest[0], &dest[10]));
  1178. // lws before and after "="
  1179. val = "attachment; filename = foo.html";
  1180. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1181. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1182. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1183. std::string(&dest[0], &dest[8]));
  1184. // lws before and after "=" with quoted-string
  1185. val = "attachment; filename = \"foo.html\"";
  1186. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1187. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1188. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1189. std::string(&dest[0], &dest[8]));
  1190. // lws after parm
  1191. val = "attachment; filename=foo.html ";
  1192. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1193. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1194. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1195. std::string(&dest[0], &dest[8]));
  1196. val = "attachment; filename=foo.html ; hello=world";
  1197. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1198. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1199. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1200. std::string(&dest[0], &dest[8]));
  1201. val = "attachment; filename=\"foo.html\" ";
  1202. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1203. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1204. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1205. std::string(&dest[0], &dest[8]));
  1206. val = "attachment; filename=\"foo.html\" ; hello=world";
  1207. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1208. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1209. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1210. std::string(&dest[0], &dest[8]));
  1211. val = "attachment; filename*=UTF-8''foo.html ; hello=world";
  1212. CPPUNIT_ASSERT_EQUAL((ssize_t)8, util::parse_content_disposition
  1213. (dest, destlen, &cs, &cslen, val.c_str(), val.size()));
  1214. CPPUNIT_ASSERT_EQUAL(std::string("foo.html"),
  1215. std::string(&dest[0], &dest[8]));
  1216. }
  1217. class Printer {
  1218. public:
  1219. template<class T>
  1220. void operator()(T t) {
  1221. std::cerr << t << ", ";
  1222. }
  1223. };
  1224. void UtilTest::testToUpper() {
  1225. std::string src = "608cabc0f2fa18c260cafd974516865c772363d5";
  1226. std::string upp = "608CABC0F2FA18C260CAFD974516865C772363D5";
  1227. CPPUNIT_ASSERT_EQUAL(upp, util::toUpper(src));
  1228. }
  1229. void UtilTest::testToLower() {
  1230. std::string src = "608CABC0F2FA18C260CAFD974516865C772363D5";
  1231. std::string upp = "608cabc0f2fa18c260cafd974516865c772363d5";
  1232. CPPUNIT_ASSERT_EQUAL(upp, util::toLower(src));
  1233. }
  1234. void UtilTest::testUppercase() {
  1235. std::string src = "608cabc0f2fa18c260cafd974516865c772363d5";
  1236. std::string ans = "608CABC0F2FA18C260CAFD974516865C772363D5";
  1237. util::uppercase(src);
  1238. CPPUNIT_ASSERT_EQUAL(ans, src);
  1239. }
  1240. void UtilTest::testLowercase() {
  1241. std::string src = "608CABC0F2FA18C260CAFD974516865C772363D5";
  1242. std::string ans = "608cabc0f2fa18c260cafd974516865c772363d5";
  1243. util::lowercase(src);
  1244. CPPUNIT_ASSERT_EQUAL(ans, src);
  1245. }
  1246. void UtilTest::testPercentDecode() {
  1247. std::string src = "http://aria2.sourceforge.net/aria2%200.7.0%20docs.html";
  1248. CPPUNIT_ASSERT_EQUAL(std::string("http://aria2.sourceforge.net/aria2 0.7.0 docs.html"),
  1249. util::percentDecode(src.begin(), src.end()));
  1250. std::string src2 = "aria2+aria2";
  1251. CPPUNIT_ASSERT_EQUAL(std::string("aria2+aria2"),
  1252. util::percentDecode(src2.begin(), src2.end()));
  1253. std::string src3 = "%5t%20";
  1254. CPPUNIT_ASSERT_EQUAL(std::string("%5t "),
  1255. util::percentDecode(src3.begin(), src3.end()));
  1256. std::string src4 = "%";
  1257. CPPUNIT_ASSERT_EQUAL(std::string("%"),
  1258. util::percentDecode(src4.begin(), src4.end()));
  1259. std::string src5 = "%3";
  1260. CPPUNIT_ASSERT_EQUAL(std::string("%3"),
  1261. util::percentDecode(src5.begin(), src5.end()));
  1262. std::string src6 = "%2f";
  1263. CPPUNIT_ASSERT_EQUAL(std::string("/"),
  1264. util::percentDecode(src6.begin(), src6.end()));
  1265. }
  1266. void UtilTest::testGetRealSize()
  1267. {
  1268. CPPUNIT_ASSERT_EQUAL((int64_t)4294967296LL, util::getRealSize("4096M"));
  1269. CPPUNIT_ASSERT_EQUAL((int64_t)1024, util::getRealSize("1K"));
  1270. CPPUNIT_ASSERT_EQUAL((int64_t)4294967296LL, util::getRealSize("4096m"));
  1271. CPPUNIT_ASSERT_EQUAL((int64_t)1024, util::getRealSize("1k"));
  1272. try {
  1273. util::getRealSize("");
  1274. CPPUNIT_FAIL("exception must be thrown.");
  1275. } catch(Exception& e) {
  1276. std::cerr << e.stackTrace();
  1277. }
  1278. try {
  1279. util::getRealSize("foo");
  1280. CPPUNIT_FAIL("exception must be thrown.");
  1281. } catch(Exception& e) {
  1282. std::cerr << e.stackTrace();
  1283. }
  1284. try {
  1285. util::getRealSize("-1");
  1286. CPPUNIT_FAIL("exception must be thrown.");
  1287. } catch(Exception& e) {
  1288. std::cerr << e.stackTrace();
  1289. }
  1290. try {
  1291. util::getRealSize("9223372036854775807K");
  1292. CPPUNIT_FAIL("exception must be thrown.");
  1293. } catch(Exception& e) {
  1294. std::cerr << e.stackTrace();
  1295. }
  1296. try {
  1297. util::getRealSize("9223372036854775807M");
  1298. CPPUNIT_FAIL("exception must be thrown.");
  1299. } catch(Exception& e) {
  1300. std::cerr << e.stackTrace();
  1301. }
  1302. }
  1303. void UtilTest::testAbbrevSize()
  1304. {
  1305. CPPUNIT_ASSERT_EQUAL(std::string("8,589,934,591Gi"),
  1306. util::abbrevSize(9223372036854775807LL));
  1307. CPPUNIT_ASSERT_EQUAL(std::string("4.0Gi"), util::abbrevSize(4294967296LL));
  1308. CPPUNIT_ASSERT_EQUAL(std::string("1.0Ki"), util::abbrevSize(1024));
  1309. CPPUNIT_ASSERT_EQUAL(std::string("0.9Ki"), util::abbrevSize(1023));
  1310. CPPUNIT_ASSERT_EQUAL(std::string("511"), util::abbrevSize(511));
  1311. CPPUNIT_ASSERT_EQUAL(std::string("0"), util::abbrevSize(0));
  1312. CPPUNIT_ASSERT_EQUAL(std::string("1.1Ki"), util::abbrevSize(1127));
  1313. CPPUNIT_ASSERT_EQUAL(std::string("1.5Mi"), util::abbrevSize(1572864));
  1314. }
  1315. void UtilTest::testToStream()
  1316. {
  1317. std::ostringstream os;
  1318. std::shared_ptr<FileEntry> f1(new FileEntry("aria2.tar.bz2", 12300, 0));
  1319. std::shared_ptr<FileEntry> f2(new FileEntry("aria2.txt", 556, 0));
  1320. std::deque<std::shared_ptr<FileEntry> > entries;
  1321. entries.push_back(f1);
  1322. entries.push_back(f2);
  1323. const char* filename = A2_TEST_OUT_DIR"/aria2_UtilTest_testToStream";
  1324. BufferedFile fp(filename, BufferedFile::WRITE);
  1325. util::toStream(entries.begin(), entries.end(), fp);
  1326. fp.close();
  1327. CPPUNIT_ASSERT_EQUAL(
  1328. std::string("Files:\n"
  1329. "idx|path/length\n"
  1330. "===+===========================================================================\n"
  1331. " 1|aria2.tar.bz2\n"
  1332. " |12KiB (12,300)\n"
  1333. "---+---------------------------------------------------------------------------\n"
  1334. " 2|aria2.txt\n"
  1335. " |556B (556)\n"
  1336. "---+---------------------------------------------------------------------------\n"),
  1337. readFile(filename));
  1338. }
  1339. void UtilTest::testIsNumber()
  1340. {
  1341. std::string s = "000";
  1342. CPPUNIT_ASSERT_EQUAL(true, util::isNumber(s.begin(),s.end()));
  1343. s = "a";
  1344. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(s.begin(), s.end()));
  1345. s = "0a";
  1346. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(s.begin(), s.end()));
  1347. s = "";
  1348. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(s.begin(), s.end()));
  1349. s = " ";
  1350. CPPUNIT_ASSERT_EQUAL(false, util::isNumber(s.begin(), s.end()));
  1351. }
  1352. void UtilTest::testIsLowercase()
  1353. {
  1354. std::string s = "alpha";
  1355. CPPUNIT_ASSERT_EQUAL(true, util::isLowercase(s.begin(), s.end()));
  1356. s = "Alpha";
  1357. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(s.begin(), s.end()));
  1358. s = "1alpha";
  1359. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(s.begin(), s.end()));
  1360. s = "";
  1361. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(s.begin(), s.end()));
  1362. s = " ";
  1363. CPPUNIT_ASSERT_EQUAL(false, util::isLowercase(s.begin(), s.end()));
  1364. }
  1365. void UtilTest::testIsUppercase()
  1366. {
  1367. std::string s = "ALPHA";
  1368. CPPUNIT_ASSERT_EQUAL(true, util::isUppercase(s.begin(), s.end()));
  1369. s = "Alpha";
  1370. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(s.begin(), s.end()));
  1371. s = "1ALPHA";
  1372. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(s.begin(), s.end()));
  1373. s = "";
  1374. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(s.begin(), s.end()));
  1375. s = " ";
  1376. CPPUNIT_ASSERT_EQUAL(false, util::isUppercase(s.begin(), s.end()));
  1377. }
  1378. void UtilTest::testMkdirs()
  1379. {
  1380. std::string dir = A2_TEST_OUT_DIR"/aria2-UtilTest-testMkdirs";
  1381. File d(dir);
  1382. if(d.exists()) {
  1383. CPPUNIT_ASSERT(d.remove());
  1384. }
  1385. CPPUNIT_ASSERT(!d.exists());
  1386. util::mkdirs(dir);
  1387. CPPUNIT_ASSERT(d.isDir());
  1388. std::string file = A2_TEST_DIR"/UtilTest.cc";
  1389. File f(file);
  1390. CPPUNIT_ASSERT(f.isFile());
  1391. try {
  1392. util::mkdirs(file);
  1393. CPPUNIT_FAIL("exception must be thrown.");
  1394. } catch(DlAbortEx& ex) {
  1395. std::cerr << ex.stackTrace() << std::endl;
  1396. }
  1397. }
  1398. void UtilTest::testConvertBitfield()
  1399. {
  1400. BitfieldMan srcBitfield(384*1024, 256*1024*256+1);
  1401. BitfieldMan destBitfield(512*1024, srcBitfield.getTotalLength());
  1402. srcBitfield.setAllBit();
  1403. srcBitfield.unsetBit(2);// <- range [768, 1152)
  1404. // which corresponds to the index [1,2] in destBitfield
  1405. util::convertBitfield(&destBitfield, &srcBitfield);
  1406. CPPUNIT_ASSERT_EQUAL(std::string("9fffffffffffffffffffffffffffffff80"),
  1407. util::toHex(destBitfield.getBitfield(),
  1408. destBitfield.getBitfieldLength()));
  1409. }
  1410. void UtilTest::testParseIntSegments()
  1411. {
  1412. {
  1413. auto sgl = util::parseIntSegments("1,3-8,10");
  1414. CPPUNIT_ASSERT(sgl.hasNext());
  1415. CPPUNIT_ASSERT_EQUAL(1, sgl.next());
  1416. CPPUNIT_ASSERT(sgl.hasNext());
  1417. CPPUNIT_ASSERT_EQUAL(3, sgl.next());
  1418. CPPUNIT_ASSERT(sgl.hasNext());
  1419. CPPUNIT_ASSERT_EQUAL(4, sgl.next());
  1420. CPPUNIT_ASSERT(sgl.hasNext());
  1421. CPPUNIT_ASSERT_EQUAL(5, sgl.next());
  1422. CPPUNIT_ASSERT(sgl.hasNext());
  1423. CPPUNIT_ASSERT_EQUAL(6, sgl.next());
  1424. CPPUNIT_ASSERT(sgl.hasNext());
  1425. CPPUNIT_ASSERT_EQUAL(7, sgl.next());
  1426. CPPUNIT_ASSERT(sgl.hasNext());
  1427. CPPUNIT_ASSERT_EQUAL(8, sgl.next());
  1428. CPPUNIT_ASSERT(sgl.hasNext());
  1429. CPPUNIT_ASSERT_EQUAL(10, sgl.next());
  1430. CPPUNIT_ASSERT(!sgl.hasNext());
  1431. CPPUNIT_ASSERT_EQUAL(0, sgl.next());
  1432. }
  1433. {
  1434. auto sgl = util::parseIntSegments(",,,1,,,3,,,");
  1435. CPPUNIT_ASSERT_EQUAL(1, sgl.next());
  1436. CPPUNIT_ASSERT_EQUAL(3, sgl.next());
  1437. CPPUNIT_ASSERT(!sgl.hasNext());
  1438. }
  1439. }
  1440. void UtilTest::testParseIntSegments_invalidRange()
  1441. {
  1442. try {
  1443. auto sgl = util::parseIntSegments("-1");
  1444. CPPUNIT_FAIL("exception must be thrown.");
  1445. } catch(Exception& e) {
  1446. }
  1447. try {
  1448. auto sgl = util::parseIntSegments("1-");
  1449. CPPUNIT_FAIL("exception must be thrown.");
  1450. } catch(Exception& e) {
  1451. }
  1452. try {
  1453. auto sgl = util::parseIntSegments("2147483648");
  1454. CPPUNIT_FAIL("exception must be thrown.");
  1455. } catch(Exception& e) {
  1456. }
  1457. try {
  1458. auto sgl = util::parseIntSegments("2147483647-2147483648");
  1459. CPPUNIT_FAIL("exception must be thrown.");
  1460. } catch(Exception& e) {
  1461. }
  1462. try {
  1463. auto sgl = util::parseIntSegments("1-2x");
  1464. CPPUNIT_FAIL("exception must be thrown.");
  1465. } catch(Exception& e) {
  1466. }
  1467. try {
  1468. auto sgl = util::parseIntSegments("3x-4");
  1469. CPPUNIT_FAIL("exception must be thrown.");
  1470. } catch(Exception& e) {
  1471. }
  1472. }
  1473. void UtilTest::testParseIntNoThrow()
  1474. {
  1475. std::string s;
  1476. int32_t n;
  1477. s = " -1 ";
  1478. CPPUNIT_ASSERT(util::parseIntNoThrow(n, s));
  1479. CPPUNIT_ASSERT_EQUAL((int32_t)-1, n);
  1480. s = "2147483647";
  1481. CPPUNIT_ASSERT(util::parseIntNoThrow(n, s));
  1482. CPPUNIT_ASSERT_EQUAL((int32_t)2147483647, n);
  1483. s = "2147483648";
  1484. CPPUNIT_ASSERT(!util::parseIntNoThrow(n, s));
  1485. s = "-2147483649";
  1486. CPPUNIT_ASSERT(!util::parseIntNoThrow(n, s));
  1487. s = "12x";
  1488. CPPUNIT_ASSERT(!util::parseIntNoThrow(n, s));
  1489. s = "";
  1490. CPPUNIT_ASSERT(!util::parseIntNoThrow(n, s));
  1491. }
  1492. void UtilTest::testParseUIntNoThrow()
  1493. {
  1494. std::string s;
  1495. uint32_t n;
  1496. s = " 2147483647 ";
  1497. CPPUNIT_ASSERT(util::parseUIntNoThrow(n, s));
  1498. CPPUNIT_ASSERT_EQUAL((uint32_t)INT32_MAX, n);
  1499. s = "2147483648";
  1500. CPPUNIT_ASSERT(!util::parseUIntNoThrow(n, s));
  1501. s = "-1";
  1502. CPPUNIT_ASSERT(!util::parseUIntNoThrow(n, s));
  1503. }
  1504. void UtilTest::testParseLLIntNoThrow()
  1505. {
  1506. std::string s;
  1507. int64_t n;
  1508. s = " 9223372036854775807 ";
  1509. CPPUNIT_ASSERT(util::parseLLIntNoThrow(n, s));
  1510. CPPUNIT_ASSERT_EQUAL((int64_t)INT64_MAX, n);
  1511. s = "9223372036854775808";
  1512. CPPUNIT_ASSERT(!util::parseLLIntNoThrow(n, s));
  1513. s = "-9223372036854775808";
  1514. CPPUNIT_ASSERT(util::parseLLIntNoThrow(n, s));
  1515. CPPUNIT_ASSERT_EQUAL((int64_t)INT64_MIN, n);
  1516. s = "-9223372036854775809";
  1517. CPPUNIT_ASSERT(!util::parseLLIntNoThrow(n, s));
  1518. }
  1519. void UtilTest::testToString_binaryStream()
  1520. {
  1521. std::shared_ptr<DiskWriter> dw(new ByteArrayDiskWriter());
  1522. std::string data(16*1024+256, 'a');
  1523. dw->initAndOpenFile();
  1524. dw->writeData((const unsigned char*)data.c_str(), data.size(), 0);
  1525. std::string readData = util::toString(dw);
  1526. CPPUNIT_ASSERT_EQUAL(data, readData);
  1527. }
  1528. void UtilTest::testItos()
  1529. {
  1530. {
  1531. int i = 0;
  1532. CPPUNIT_ASSERT_EQUAL(std::string("0"), util::itos(i));
  1533. }
  1534. {
  1535. int i = 100;
  1536. CPPUNIT_ASSERT_EQUAL(std::string("100"), util::itos(i, true));
  1537. }
  1538. {
  1539. int i = 100;
  1540. CPPUNIT_ASSERT_EQUAL(std::string("100"), util::itos(i));
  1541. }
  1542. {
  1543. int i = 12345;
  1544. CPPUNIT_ASSERT_EQUAL(std::string("12,345"), util::itos(i, true));
  1545. }
  1546. {
  1547. int i = 12345;
  1548. CPPUNIT_ASSERT_EQUAL(std::string("12345"), util::itos(i));
  1549. }
  1550. {
  1551. int i = -12345;
  1552. CPPUNIT_ASSERT_EQUAL(std::string("-12,345"), util::itos(i, true));
  1553. }
  1554. {
  1555. int64_t i = INT64_MAX;
  1556. CPPUNIT_ASSERT_EQUAL(std::string("9,223,372,036,854,775,807"),
  1557. util::itos(i, true));
  1558. }
  1559. {
  1560. int64_t i = INT64_MIN;
  1561. CPPUNIT_ASSERT_EQUAL(std::string("-9,223,372,036,854,775,808"),
  1562. util::itos(i, true));
  1563. }
  1564. }
  1565. void UtilTest::testUitos()
  1566. {
  1567. {
  1568. uint16_t i = 12345;
  1569. CPPUNIT_ASSERT_EQUAL(std::string("12345"), util::uitos(i));
  1570. }
  1571. {
  1572. int16_t i = -12345;
  1573. CPPUNIT_ASSERT_EQUAL(std::string("/.-,+"), util::uitos(i));
  1574. }
  1575. }
  1576. void UtilTest::testNtoh64()
  1577. {
  1578. uint64_t x = 0xff00ff00ee00ee00LL;
  1579. #ifdef WORDS_BIGENDIAN
  1580. CPPUNIT_ASSERT_EQUAL(x, ntoh64(x));
  1581. CPPUNIT_ASSERT_EQUAL(x, hton64(x));
  1582. #else // !WORDS_BIGENDIAN
  1583. uint64_t y = 0x00ee00ee00ff00ffLL;
  1584. CPPUNIT_ASSERT_EQUAL(y, ntoh64(x));
  1585. CPPUNIT_ASSERT_EQUAL(x, hton64(y));
  1586. #endif // !WORDS_BIGENDIAN
  1587. }
  1588. void UtilTest::testPercentEncode()
  1589. {
  1590. CPPUNIT_ASSERT_EQUAL
  1591. (std::string("%3A%2F%3F%23%5B%5D%40%21%25%26%27%28%29%2A%2B%2C%3B%3D"),
  1592. util::percentEncode(":/?#[]@!%&'()*+,;="));
  1593. std::string unreserved =
  1594. "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  1595. "abcdefghijklmnopqrstuvwxyz"
  1596. "0123456789"
  1597. "-._~";
  1598. CPPUNIT_ASSERT_EQUAL(unreserved, util::percentEncode(unreserved));
  1599. CPPUNIT_ASSERT_EQUAL(std::string("1%5EA%20"), util::percentEncode("1^A "));
  1600. }
  1601. void UtilTest::testPercentEncodeMini()
  1602. {
  1603. CPPUNIT_ASSERT_EQUAL(std::string("%80"),
  1604. util::percentEncodeMini({(char)0x80}));
  1605. }
  1606. void UtilTest::testHtmlEscape()
  1607. {
  1608. CPPUNIT_ASSERT_EQUAL(std::string("aria2&lt;&gt;&quot;&#39;util"),
  1609. util::htmlEscape("aria2<>\"'util"));
  1610. }
  1611. void UtilTest::testJoinPath()
  1612. {
  1613. const std::string dir1dir2file[] = { "dir1", "dir2", "file" };
  1614. CPPUNIT_ASSERT_EQUAL
  1615. (std::string("dir1/dir2/file"),
  1616. util::joinPath(std::begin(dir1dir2file), std::end(dir1dir2file)));
  1617. const std::string dirparentfile[] = { "dir", "..", "file" };
  1618. CPPUNIT_ASSERT_EQUAL
  1619. (std::string("file"),
  1620. util::joinPath(std::begin(dirparentfile), std::end(dirparentfile)));
  1621. const std::string dirparentparentfile[] = { "dir", "..", "..", "file" };
  1622. CPPUNIT_ASSERT_EQUAL
  1623. (std::string("file"),
  1624. util::joinPath(std::begin(dirparentparentfile),
  1625. std::end(dirparentparentfile)));
  1626. const std::string dirdotfile[] = { "dir", ".", "file" };
  1627. CPPUNIT_ASSERT_EQUAL(std::string("dir/file"),
  1628. util::joinPath(std::begin(dirdotfile),
  1629. std::end(dirdotfile)));
  1630. const std::string empty[] = {};
  1631. CPPUNIT_ASSERT_EQUAL(std::string(""), util::joinPath(&empty[0], &empty[0]));
  1632. const std::string parentdot[] = { "..", "." };
  1633. CPPUNIT_ASSERT_EQUAL(std::string(""),
  1634. util::joinPath(std::begin(parentdot),
  1635. std::end(parentdot)));
  1636. }
  1637. void UtilTest::testParseIndexPath()
  1638. {
  1639. std::pair<size_t, std::string> p = util::parseIndexPath("1=foo");
  1640. CPPUNIT_ASSERT_EQUAL((size_t)1, p.first);
  1641. CPPUNIT_ASSERT_EQUAL(std::string("foo"), p.second);
  1642. try {
  1643. util::parseIndexPath("1X=foo");
  1644. CPPUNIT_FAIL("exception must be thrown.");
  1645. } catch(Exception& e) {
  1646. // success
  1647. }
  1648. try {
  1649. util::parseIndexPath("1=");
  1650. CPPUNIT_FAIL("exception must be thrown.");
  1651. } catch(Exception& e) {
  1652. // success
  1653. }
  1654. }
  1655. void UtilTest::testCreateIndexPaths()
  1656. {
  1657. std::stringstream in
  1658. ("1=/tmp/myfile\n"
  1659. "100=/myhome/mypicture.png\n");
  1660. std::vector<std::pair<size_t, std::string> > m = util::createIndexPaths(in);
  1661. CPPUNIT_ASSERT_EQUAL((size_t)2, m.size());
  1662. CPPUNIT_ASSERT_EQUAL((size_t)1, m[0].first);
  1663. CPPUNIT_ASSERT_EQUAL(std::string("/tmp/myfile"), m[0].second);
  1664. CPPUNIT_ASSERT_EQUAL((size_t)100, m[1].first);
  1665. CPPUNIT_ASSERT_EQUAL(std::string("/myhome/mypicture.png"), m[1].second);
  1666. }
  1667. void UtilTest::testGenerateRandomData()
  1668. {
  1669. using namespace std;
  1670. // Simple sanity check
  1671. unsigned char data1[25];
  1672. memset(data1, 0, sizeof(data1));
  1673. util::generateRandomData(data1, sizeof(data1));
  1674. unsigned char data2[25];
  1675. memset(data2, 0, sizeof(data2));
  1676. util::generateRandomData(data2, sizeof(data2));
  1677. CPPUNIT_ASSERT(memcmp(data1, data2, sizeof(data1)) != 0);
  1678. // Simple stddev/mean tests
  1679. map<uint8_t, size_t> counts;
  1680. uint8_t bytes[1 << 20];
  1681. for (auto i = 0; i < 10; ++i) {
  1682. util::generateRandomData(bytes, sizeof(bytes));
  1683. for (auto b : bytes) {
  1684. counts[b]++;
  1685. }
  1686. }
  1687. CPPUNIT_ASSERT_MESSAGE("Should see all kinds of bytes", counts.size() == 256);
  1688. if (counts.size() != 256) {
  1689. throw std::domain_error(
  1690. "Would have expected to see at one of each possible byte value!");
  1691. }
  1692. double sum = accumulate(
  1693. counts.begin(),
  1694. counts.end(),
  1695. 0.0,
  1696. [](double total, const decltype(counts)::value_type & elem) {
  1697. return total + elem.second;
  1698. });
  1699. double mean = sum / counts.size();
  1700. vector<double> diff(counts.size());
  1701. transform(
  1702. counts.begin(),
  1703. counts.end(),
  1704. diff.begin(),
  1705. [&](const decltype(counts)::value_type & elem) -> double {
  1706. return (double)elem.second - mean;
  1707. });
  1708. double sq_sum = inner_product(diff.begin(), diff.end(), diff.begin(), 0.0);
  1709. double stddev = sqrt(sq_sum / counts.size());
  1710. cout << "stddev: " << fixed << stddev << endl;
  1711. CPPUNIT_ASSERT_MESSAGE("stddev makes sense (lower)", stddev <= 320);
  1712. CPPUNIT_ASSERT_MESSAGE("stddev makes sense (upper)", stddev >= 100);
  1713. }
  1714. void UtilTest::testFromHex()
  1715. {
  1716. std::string src;
  1717. std::string dest;
  1718. src = "0011fF";
  1719. dest = util::fromHex(src.begin(), src.end());
  1720. CPPUNIT_ASSERT_EQUAL((size_t)3, dest.size());
  1721. CPPUNIT_ASSERT_EQUAL((char)0x00, dest[0]);
  1722. CPPUNIT_ASSERT_EQUAL((char)0x11, dest[1]);
  1723. CPPUNIT_ASSERT_EQUAL((char)0xff, dest[2]);
  1724. src = "0011f";
  1725. CPPUNIT_ASSERT(util::fromHex(src.begin(), src.end()).empty());
  1726. src = "001g";
  1727. CPPUNIT_ASSERT(util::fromHex(src.begin(), src.end()).empty());
  1728. }
  1729. void UtilTest::testParsePrioritizePieceRange()
  1730. {
  1731. // piece index
  1732. // 0 1 2 3 4 5 6 7
  1733. // | | | |
  1734. // file1 | | |
  1735. // | | |
  1736. // file2 | |
  1737. // file3 |
  1738. // | |
  1739. // file4 |
  1740. size_t pieceLength = 1024;
  1741. std::vector<std::shared_ptr<FileEntry> > entries(4, std::shared_ptr<FileEntry>());
  1742. entries[0].reset(new FileEntry("file1", 1024, 0));
  1743. entries[1].reset(new FileEntry("file2",2560,entries[0]->getLastOffset()));
  1744. entries[2].reset(new FileEntry("file3",0,entries[1]->getLastOffset()));
  1745. entries[3].reset(new FileEntry("file4",3584,entries[2]->getLastOffset()));
  1746. std::vector<size_t> result;
  1747. util::parsePrioritizePieceRange(result, "head=1", entries, pieceLength);
  1748. CPPUNIT_ASSERT_EQUAL((size_t)3, result.size());
  1749. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  1750. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  1751. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  1752. result.clear();
  1753. util::parsePrioritizePieceRange(result, "tail=1", entries, pieceLength);
  1754. CPPUNIT_ASSERT_EQUAL((size_t)3, result.size());
  1755. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  1756. CPPUNIT_ASSERT_EQUAL((size_t)3, result[1]);
  1757. CPPUNIT_ASSERT_EQUAL((size_t)6, result[2]);
  1758. result.clear();
  1759. util::parsePrioritizePieceRange(result, "head=1K", entries, pieceLength);
  1760. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  1761. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  1762. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  1763. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  1764. CPPUNIT_ASSERT_EQUAL((size_t)4, result[3]);
  1765. result.clear();
  1766. util::parsePrioritizePieceRange(result, "head", entries, pieceLength, 1024);
  1767. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  1768. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  1769. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  1770. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  1771. CPPUNIT_ASSERT_EQUAL((size_t)4, result[3]);
  1772. result.clear();
  1773. util::parsePrioritizePieceRange(result, "tail=1K", entries, pieceLength);
  1774. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  1775. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  1776. CPPUNIT_ASSERT_EQUAL((size_t)2, result[1]);
  1777. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  1778. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  1779. result.clear();
  1780. util::parsePrioritizePieceRange(result, "tail", entries, pieceLength, 1024);
  1781. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  1782. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  1783. CPPUNIT_ASSERT_EQUAL((size_t)2, result[1]);
  1784. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  1785. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  1786. result.clear();
  1787. util::parsePrioritizePieceRange(result, "head=1,tail=1", entries, pieceLength);
  1788. CPPUNIT_ASSERT_EQUAL((size_t)4, result.size());
  1789. CPPUNIT_ASSERT_EQUAL((size_t)0, result[0]);
  1790. CPPUNIT_ASSERT_EQUAL((size_t)1, result[1]);
  1791. CPPUNIT_ASSERT_EQUAL((size_t)3, result[2]);
  1792. CPPUNIT_ASSERT_EQUAL((size_t)6, result[3]);
  1793. result.clear();
  1794. util::parsePrioritizePieceRange(result, "head=300M,tail=300M",
  1795. entries, pieceLength);
  1796. CPPUNIT_ASSERT_EQUAL((size_t)7, result.size());
  1797. for(size_t i = 0; i < 7; ++i) {
  1798. CPPUNIT_ASSERT_EQUAL(i, result[i]);
  1799. }
  1800. result.clear();
  1801. util::parsePrioritizePieceRange(result, "", entries, pieceLength);
  1802. CPPUNIT_ASSERT(result.empty());
  1803. }
  1804. void UtilTest::testApplyDir()
  1805. {
  1806. CPPUNIT_ASSERT_EQUAL(std::string("./pred"), util::applyDir("", "pred"));
  1807. CPPUNIT_ASSERT_EQUAL(std::string("/pred"), util::applyDir("/", "pred"));
  1808. CPPUNIT_ASSERT_EQUAL(std::string("./pred"), util::applyDir(".", "pred"));
  1809. CPPUNIT_ASSERT_EQUAL(std::string("/dl/pred"), util::applyDir("/dl", "pred"));
  1810. }
  1811. void UtilTest::testFixTaintedBasename()
  1812. {
  1813. CPPUNIT_ASSERT_EQUAL(std::string("a%2Fb"), util::fixTaintedBasename("a/b"));
  1814. #ifdef __MINGW32__
  1815. CPPUNIT_ASSERT_EQUAL(std::string("a%5Cb"), util::fixTaintedBasename("a\\b"));
  1816. #else // !__MINGW32__
  1817. CPPUNIT_ASSERT_EQUAL(std::string("a\\b"), util::fixTaintedBasename("a\\b"));
  1818. #endif // !__MINGW32__
  1819. }
  1820. void UtilTest::testIsNumericHost()
  1821. {
  1822. CPPUNIT_ASSERT(util::isNumericHost("192.168.0.1"));
  1823. CPPUNIT_ASSERT(!util::isNumericHost("aria2.sf.net"));
  1824. CPPUNIT_ASSERT(util::isNumericHost("::1"));
  1825. }
  1826. void UtilTest::testDetectDirTraversal()
  1827. {
  1828. CPPUNIT_ASSERT(util::detectDirTraversal("/foo"));
  1829. CPPUNIT_ASSERT(util::detectDirTraversal("./foo"));
  1830. CPPUNIT_ASSERT(util::detectDirTraversal("../foo"));
  1831. CPPUNIT_ASSERT(util::detectDirTraversal("foo/../bar"));
  1832. CPPUNIT_ASSERT(util::detectDirTraversal("foo/./bar"));
  1833. CPPUNIT_ASSERT(util::detectDirTraversal("foo/."));
  1834. CPPUNIT_ASSERT(util::detectDirTraversal("foo/.."));
  1835. CPPUNIT_ASSERT(util::detectDirTraversal("."));
  1836. CPPUNIT_ASSERT(util::detectDirTraversal(".."));
  1837. CPPUNIT_ASSERT(util::detectDirTraversal("/"));
  1838. CPPUNIT_ASSERT(util::detectDirTraversal("foo/"));
  1839. CPPUNIT_ASSERT(util::detectDirTraversal("\t"));
  1840. CPPUNIT_ASSERT(!util::detectDirTraversal("foo/bar"));
  1841. CPPUNIT_ASSERT(!util::detectDirTraversal("foo"));
  1842. }
  1843. void UtilTest::testEscapePath()
  1844. {
  1845. CPPUNIT_ASSERT_EQUAL(std::string("foo%00bar%00%01"),
  1846. util::escapePath(std::string("foo")+(char)0x00+
  1847. std::string("bar")+(char)0x00+
  1848. (char)0x01));
  1849. #ifdef __MINGW32__
  1850. CPPUNIT_ASSERT_EQUAL(std::string("foo%5Cbar"), util::escapePath("foo\\bar"));
  1851. #else // !__MINGW32__
  1852. CPPUNIT_ASSERT_EQUAL(std::string("foo\\bar"), util::escapePath("foo\\bar"));
  1853. #endif // !__MINGW32__
  1854. }
  1855. void UtilTest::testInSameCidrBlock()
  1856. {
  1857. CPPUNIT_ASSERT(util::inSameCidrBlock("192.168.128.1", "192.168.0.1", 16));
  1858. CPPUNIT_ASSERT(!util::inSameCidrBlock("192.168.128.1", "192.168.0.1", 17));
  1859. CPPUNIT_ASSERT(util::inSameCidrBlock("192.168.0.1", "192.168.0.1", 32));
  1860. CPPUNIT_ASSERT(!util::inSameCidrBlock("192.168.0.1", "192.168.0.0", 32));
  1861. CPPUNIT_ASSERT(util::inSameCidrBlock("192.168.0.1", "10.0.0.1", 0));
  1862. CPPUNIT_ASSERT(util::inSameCidrBlock("2001:db8::2:1", "2001:db0::2:2", 28));
  1863. CPPUNIT_ASSERT(!util::inSameCidrBlock("2001:db8::2:1", "2001:db0::2:2", 29));
  1864. CPPUNIT_ASSERT(!util::inSameCidrBlock("2001:db8::2:1", "192.168.0.1", 8));
  1865. }
  1866. void UtilTest::testIsUtf8String()
  1867. {
  1868. CPPUNIT_ASSERT(util::isUtf8("ascii"));
  1869. // "Hello World" in Japanese UTF-8
  1870. CPPUNIT_ASSERT(util::isUtf8
  1871. (fromHex("e38193e38293e381abe381a1e381afe4b896e7958c")));
  1872. // "World" in Shift_JIS
  1873. CPPUNIT_ASSERT(!util::isUtf8(fromHex("90a28a")+"E"));
  1874. // UTF8-2
  1875. CPPUNIT_ASSERT(util::isUtf8(fromHex("c280")));
  1876. CPPUNIT_ASSERT(util::isUtf8(fromHex("dfbf")));
  1877. // UTF8-3
  1878. CPPUNIT_ASSERT(util::isUtf8(fromHex("e0a080")));
  1879. CPPUNIT_ASSERT(util::isUtf8(fromHex("e0bf80")));
  1880. CPPUNIT_ASSERT(util::isUtf8(fromHex("e18080")));
  1881. CPPUNIT_ASSERT(util::isUtf8(fromHex("ec8080")));
  1882. CPPUNIT_ASSERT(util::isUtf8(fromHex("ed8080")));
  1883. CPPUNIT_ASSERT(util::isUtf8(fromHex("ed9f80")));
  1884. CPPUNIT_ASSERT(util::isUtf8(fromHex("ee8080")));
  1885. CPPUNIT_ASSERT(util::isUtf8(fromHex("ef8080")));
  1886. // UTF8-4
  1887. CPPUNIT_ASSERT(util::isUtf8(fromHex("f0908080")));
  1888. CPPUNIT_ASSERT(util::isUtf8(fromHex("f0bf8080")));
  1889. CPPUNIT_ASSERT(util::isUtf8(fromHex("f1808080")));
  1890. CPPUNIT_ASSERT(util::isUtf8(fromHex("f3808080")));
  1891. CPPUNIT_ASSERT(util::isUtf8(fromHex("f4808080")));
  1892. CPPUNIT_ASSERT(util::isUtf8(fromHex("f48f8080")));
  1893. CPPUNIT_ASSERT(util::isUtf8(""));
  1894. CPPUNIT_ASSERT(!util::isUtf8(fromHex("00")));
  1895. }
  1896. void UtilTest::testNextParam()
  1897. {
  1898. std::string s1 = " :a : b=c :d=b::::g::";
  1899. std::pair<std::string::iterator, bool> r;
  1900. std::string name, value;
  1901. r = util::nextParam(name, value, s1.begin(), s1.end(), ':');
  1902. CPPUNIT_ASSERT(r.second);
  1903. CPPUNIT_ASSERT_EQUAL(std::string("a"), name);
  1904. CPPUNIT_ASSERT_EQUAL(std::string(), value);
  1905. r = util::nextParam(name, value, r.first, s1.end(), ':');
  1906. CPPUNIT_ASSERT(r.second);
  1907. CPPUNIT_ASSERT_EQUAL(std::string("b"), name);
  1908. CPPUNIT_ASSERT_EQUAL(std::string("c"), value);
  1909. r = util::nextParam(name, value, r.first, s1.end(), ':');
  1910. CPPUNIT_ASSERT(r.second);
  1911. CPPUNIT_ASSERT_EQUAL(std::string("d"), name);
  1912. CPPUNIT_ASSERT_EQUAL(std::string("b"), value);
  1913. r = util::nextParam(name, value, r.first, s1.end(), ':');
  1914. CPPUNIT_ASSERT(r.second);
  1915. CPPUNIT_ASSERT_EQUAL(std::string("g"), name);
  1916. CPPUNIT_ASSERT_EQUAL(std::string(), value);
  1917. std::string s2 = "";
  1918. r = util::nextParam(name, value, s2.begin(), s2.end(), ':');
  1919. CPPUNIT_ASSERT(!r.second);
  1920. std::string s3 = " ";
  1921. r = util::nextParam(name, value, s3.begin(), s3.end(), ':');
  1922. CPPUNIT_ASSERT(!r.second);
  1923. std::string s4 = ":::";
  1924. r = util::nextParam(name, value, s4.begin(), s4.end(), ':');
  1925. CPPUNIT_ASSERT(!r.second);
  1926. }
  1927. void UtilTest::testNoProxyDomainMatch()
  1928. {
  1929. CPPUNIT_ASSERT(util::noProxyDomainMatch("localhost", "localhost"));
  1930. CPPUNIT_ASSERT(util::noProxyDomainMatch("192.168.0.1", "192.168.0.1"));
  1931. CPPUNIT_ASSERT(util::noProxyDomainMatch("www.example.org", ".example.org"));
  1932. CPPUNIT_ASSERT(!util::noProxyDomainMatch("www.example.org", "example.org"));
  1933. CPPUNIT_ASSERT(!util::noProxyDomainMatch("192.168.0.1", "0.1"));
  1934. CPPUNIT_ASSERT(!util::noProxyDomainMatch("example.org", "example.com"));
  1935. CPPUNIT_ASSERT(!util::noProxyDomainMatch("example.org", "www.example.org"));
  1936. }
  1937. void UtilTest::testInPrivateAddress()
  1938. {
  1939. CPPUNIT_ASSERT(!util::inPrivateAddress("localhost"));
  1940. CPPUNIT_ASSERT(util::inPrivateAddress("192.168.0.1"));
  1941. // Only checks prefix..
  1942. CPPUNIT_ASSERT(util::inPrivateAddress("10."));
  1943. CPPUNIT_ASSERT(!util::inPrivateAddress("172."));
  1944. CPPUNIT_ASSERT(!util::inPrivateAddress("172.15.0.0"));
  1945. CPPUNIT_ASSERT(util::inPrivateAddress("172.16.0.0"));
  1946. CPPUNIT_ASSERT(util::inPrivateAddress("172.31.0.0"));
  1947. CPPUNIT_ASSERT(!util::inPrivateAddress("172.32.0.0"));
  1948. }
  1949. void UtilTest::testSecfmt()
  1950. {
  1951. CPPUNIT_ASSERT_EQUAL(std::string("0s"), util::secfmt(0));
  1952. CPPUNIT_ASSERT_EQUAL(std::string("1s"), util::secfmt(1));
  1953. CPPUNIT_ASSERT_EQUAL(std::string("9s"), util::secfmt(9));
  1954. CPPUNIT_ASSERT_EQUAL(std::string("10s"), util::secfmt(10));
  1955. CPPUNIT_ASSERT_EQUAL(std::string("1m"), util::secfmt(60));
  1956. CPPUNIT_ASSERT_EQUAL(std::string("1m59s"), util::secfmt(119));
  1957. CPPUNIT_ASSERT_EQUAL(std::string("2m"), util::secfmt(120));
  1958. CPPUNIT_ASSERT_EQUAL(std::string("59m59s"), util::secfmt(3599));
  1959. CPPUNIT_ASSERT_EQUAL(std::string("1h"), util::secfmt(3600));
  1960. }
  1961. void UtilTest::testTlsHostnameMatch()
  1962. {
  1963. CPPUNIT_ASSERT(util::tlsHostnameMatch("Foo.com", "foo.com"));
  1964. CPPUNIT_ASSERT(util::tlsHostnameMatch("*.a.com", "foo.a.com"));
  1965. CPPUNIT_ASSERT(!util::tlsHostnameMatch("*.a.com", "bar.foo.a.com"));
  1966. CPPUNIT_ASSERT(!util::tlsHostnameMatch("f*.com", "foo.com"));
  1967. CPPUNIT_ASSERT(!util::tlsHostnameMatch("*.com", "bar.com"));
  1968. CPPUNIT_ASSERT(util::tlsHostnameMatch("com", "com"));
  1969. CPPUNIT_ASSERT(!util::tlsHostnameMatch("foo.*", "foo.com"));
  1970. CPPUNIT_ASSERT(util::tlsHostnameMatch("a.foo.com", "A.foo.com"));
  1971. CPPUNIT_ASSERT(!util::tlsHostnameMatch("a.foo.com", "b.foo.com"));
  1972. CPPUNIT_ASSERT(!util::tlsHostnameMatch("*a.foo.com", "a.foo.com"));
  1973. CPPUNIT_ASSERT(util::tlsHostnameMatch("*a.foo.com", "ba.foo.com"));
  1974. CPPUNIT_ASSERT(!util::tlsHostnameMatch("a*.foo.com", "a.foo.com"));
  1975. CPPUNIT_ASSERT(util::tlsHostnameMatch("a*.foo.com", "ab.foo.com"));
  1976. CPPUNIT_ASSERT(!util::tlsHostnameMatch("foo.b*z.foo.com", "foo.baz.foo.com"));
  1977. CPPUNIT_ASSERT(util::tlsHostnameMatch("B*z.foo.com", "bAZ.Foo.com"));
  1978. CPPUNIT_ASSERT(!util::tlsHostnameMatch("b*z.foo.com", "bz.foo.com"));
  1979. CPPUNIT_ASSERT(!util::tlsHostnameMatch("*", "foo"));
  1980. CPPUNIT_ASSERT(!util::tlsHostnameMatch("*", ""));
  1981. CPPUNIT_ASSERT(util::tlsHostnameMatch("", ""));
  1982. CPPUNIT_ASSERT(!util::tlsHostnameMatch("xn--*.a.b", "xn--c.a.b"));
  1983. }
  1984. } // namespace aria2