string_piece_unittest.cc 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689
  1. // Copyright (c) 2012 The Chromium Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style license that can be
  3. // found in the LICENSE file.
  4. #include <string>
  5. #include "butil/strings/string16.h"
  6. #include "butil/strings/string_piece.h"
  7. #include "butil/strings/utf_string_conversions.h"
  8. #include <gtest/gtest.h>
  9. namespace butil {
  10. template <typename T>
  11. class CommonStringPieceTest : public ::testing::Test {
  12. public:
  13. static const T as_string(const char* input) {
  14. return T(input);
  15. }
  16. static const T& as_string(const T& input) {
  17. return input;
  18. }
  19. };
  20. template <>
  21. class CommonStringPieceTest<string16> : public ::testing::Test {
  22. public:
  23. static const string16 as_string(const char* input) {
  24. return ASCIIToUTF16(input);
  25. }
  26. static const string16 as_string(const std::string& input) {
  27. return ASCIIToUTF16(input);
  28. }
  29. };
  30. typedef ::testing::Types<std::string, string16> SupportedStringTypes;
  31. TYPED_TEST_CASE(CommonStringPieceTest, SupportedStringTypes);
  32. TYPED_TEST(CommonStringPieceTest, CheckComparisonOperators) {
  33. #define CMP_Y(op, x, y) \
  34. { \
  35. TypeParam lhs(TestFixture::as_string(x)); \
  36. TypeParam rhs(TestFixture::as_string(y)); \
  37. ASSERT_TRUE( (BasicStringPiece<TypeParam>((lhs.c_str())) op \
  38. BasicStringPiece<TypeParam>((rhs.c_str())))); \
  39. ASSERT_TRUE( (BasicStringPiece<TypeParam>((lhs.c_str())).compare( \
  40. BasicStringPiece<TypeParam>((rhs.c_str()))) op 0)); \
  41. }
  42. #define CMP_N(op, x, y) \
  43. { \
  44. TypeParam lhs(TestFixture::as_string(x)); \
  45. TypeParam rhs(TestFixture::as_string(y)); \
  46. ASSERT_FALSE( (BasicStringPiece<TypeParam>((lhs.c_str())) op \
  47. BasicStringPiece<TypeParam>((rhs.c_str())))); \
  48. ASSERT_FALSE( (BasicStringPiece<TypeParam>((lhs.c_str())).compare( \
  49. BasicStringPiece<TypeParam>((rhs.c_str()))) op 0)); \
  50. }
  51. CMP_Y(==, "", "");
  52. CMP_Y(==, "a", "a");
  53. CMP_Y(==, "aa", "aa");
  54. CMP_N(==, "a", "");
  55. CMP_N(==, "", "a");
  56. CMP_N(==, "a", "b");
  57. CMP_N(==, "a", "aa");
  58. CMP_N(==, "aa", "a");
  59. CMP_N(!=, "", "");
  60. CMP_N(!=, "a", "a");
  61. CMP_N(!=, "aa", "aa");
  62. CMP_Y(!=, "a", "");
  63. CMP_Y(!=, "", "a");
  64. CMP_Y(!=, "a", "b");
  65. CMP_Y(!=, "a", "aa");
  66. CMP_Y(!=, "aa", "a");
  67. CMP_Y(<, "a", "b");
  68. CMP_Y(<, "a", "aa");
  69. CMP_Y(<, "aa", "b");
  70. CMP_Y(<, "aa", "bb");
  71. CMP_N(<, "a", "a");
  72. CMP_N(<, "b", "a");
  73. CMP_N(<, "aa", "a");
  74. CMP_N(<, "b", "aa");
  75. CMP_N(<, "bb", "aa");
  76. CMP_Y(<=, "a", "a");
  77. CMP_Y(<=, "a", "b");
  78. CMP_Y(<=, "a", "aa");
  79. CMP_Y(<=, "aa", "b");
  80. CMP_Y(<=, "aa", "bb");
  81. CMP_N(<=, "b", "a");
  82. CMP_N(<=, "aa", "a");
  83. CMP_N(<=, "b", "aa");
  84. CMP_N(<=, "bb", "aa");
  85. CMP_N(>=, "a", "b");
  86. CMP_N(>=, "a", "aa");
  87. CMP_N(>=, "aa", "b");
  88. CMP_N(>=, "aa", "bb");
  89. CMP_Y(>=, "a", "a");
  90. CMP_Y(>=, "b", "a");
  91. CMP_Y(>=, "aa", "a");
  92. CMP_Y(>=, "b", "aa");
  93. CMP_Y(>=, "bb", "aa");
  94. CMP_N(>, "a", "a");
  95. CMP_N(>, "a", "b");
  96. CMP_N(>, "a", "aa");
  97. CMP_N(>, "aa", "b");
  98. CMP_N(>, "aa", "bb");
  99. CMP_Y(>, "b", "a");
  100. CMP_Y(>, "aa", "a");
  101. CMP_Y(>, "b", "aa");
  102. CMP_Y(>, "bb", "aa");
  103. std::string x;
  104. for (int i = 0; i < 256; i++) {
  105. x += 'a';
  106. std::string y = x;
  107. CMP_Y(==, x, y);
  108. for (int j = 0; j < i; j++) {
  109. std::string z = x;
  110. z[j] = 'b'; // Differs in position 'j'
  111. CMP_N(==, x, z);
  112. }
  113. }
  114. #undef CMP_Y
  115. #undef CMP_N
  116. }
  117. TYPED_TEST(CommonStringPieceTest, CheckSTL) {
  118. TypeParam alphabet(TestFixture::as_string("abcdefghijklmnopqrstuvwxyz"));
  119. TypeParam abc(TestFixture::as_string("abc"));
  120. TypeParam xyz(TestFixture::as_string("xyz"));
  121. TypeParam foobar(TestFixture::as_string("foobar"));
  122. BasicStringPiece<TypeParam> a(alphabet);
  123. BasicStringPiece<TypeParam> b(abc);
  124. BasicStringPiece<TypeParam> c(xyz);
  125. BasicStringPiece<TypeParam> d(foobar);
  126. BasicStringPiece<TypeParam> e;
  127. TypeParam temp(TestFixture::as_string("123"));
  128. temp += static_cast<typename TypeParam::value_type>(0);
  129. temp += TestFixture::as_string("456");
  130. BasicStringPiece<TypeParam> f(temp);
  131. ASSERT_EQ(a[6], static_cast<typename TypeParam::value_type>('g'));
  132. ASSERT_EQ(b[0], static_cast<typename TypeParam::value_type>('a'));
  133. ASSERT_EQ(c[2], static_cast<typename TypeParam::value_type>('z'));
  134. ASSERT_EQ(f[3], static_cast<typename TypeParam::value_type>('\0'));
  135. ASSERT_EQ(f[5], static_cast<typename TypeParam::value_type>('5'));
  136. ASSERT_EQ(*d.data(), static_cast<typename TypeParam::value_type>('f'));
  137. ASSERT_EQ(d.data()[5], static_cast<typename TypeParam::value_type>('r'));
  138. ASSERT_TRUE(e.data() == NULL);
  139. ASSERT_EQ(*a.begin(), static_cast<typename TypeParam::value_type>('a'));
  140. ASSERT_EQ(*(b.begin() + 2), static_cast<typename TypeParam::value_type>('c'));
  141. ASSERT_EQ(*(c.end() - 1), static_cast<typename TypeParam::value_type>('z'));
  142. ASSERT_EQ(*a.rbegin(), static_cast<typename TypeParam::value_type>('z'));
  143. ASSERT_EQ(*(b.rbegin() + 2),
  144. static_cast<typename TypeParam::value_type>('a'));
  145. ASSERT_EQ(*(c.rend() - 1), static_cast<typename TypeParam::value_type>('x'));
  146. ASSERT_TRUE(a.rbegin() + 26 == a.rend());
  147. ASSERT_EQ(a.size(), 26U);
  148. ASSERT_EQ(b.size(), 3U);
  149. ASSERT_EQ(c.size(), 3U);
  150. ASSERT_EQ(d.size(), 6U);
  151. ASSERT_EQ(e.size(), 0U);
  152. ASSERT_EQ(f.size(), 7U);
  153. ASSERT_TRUE(!d.empty());
  154. ASSERT_TRUE(d.begin() != d.end());
  155. ASSERT_TRUE(d.begin() + 6 == d.end());
  156. ASSERT_TRUE(e.empty());
  157. ASSERT_TRUE(e.begin() == e.end());
  158. d.clear();
  159. ASSERT_EQ(d.size(), 0U);
  160. ASSERT_TRUE(d.empty());
  161. ASSERT_TRUE(d.data() == NULL);
  162. ASSERT_TRUE(d.begin() == d.end());
  163. ASSERT_GE(a.max_size(), a.capacity());
  164. ASSERT_GE(a.capacity(), a.size());
  165. }
  166. TYPED_TEST(CommonStringPieceTest, CheckFind) {
  167. typedef BasicStringPiece<TypeParam> Piece;
  168. TypeParam alphabet(TestFixture::as_string("abcdefghijklmnopqrstuvwxyz"));
  169. TypeParam abc(TestFixture::as_string("abc"));
  170. TypeParam xyz(TestFixture::as_string("xyz"));
  171. TypeParam foobar(TestFixture::as_string("foobar"));
  172. BasicStringPiece<TypeParam> a(alphabet);
  173. BasicStringPiece<TypeParam> b(abc);
  174. BasicStringPiece<TypeParam> c(xyz);
  175. BasicStringPiece<TypeParam> d(foobar);
  176. d.clear();
  177. Piece e;
  178. TypeParam temp(TestFixture::as_string("123"));
  179. temp.push_back('\0');
  180. temp += TestFixture::as_string("456");
  181. Piece f(temp);
  182. typename TypeParam::value_type buf[4] = { '%', '%', '%', '%' };
  183. ASSERT_EQ(a.copy(buf, 4), 4U);
  184. ASSERT_EQ(buf[0], a[0]);
  185. ASSERT_EQ(buf[1], a[1]);
  186. ASSERT_EQ(buf[2], a[2]);
  187. ASSERT_EQ(buf[3], a[3]);
  188. ASSERT_EQ(a.copy(buf, 3, 7), 3U);
  189. ASSERT_EQ(buf[0], a[7]);
  190. ASSERT_EQ(buf[1], a[8]);
  191. ASSERT_EQ(buf[2], a[9]);
  192. ASSERT_EQ(buf[3], a[3]);
  193. ASSERT_EQ(c.copy(buf, 99), 3U);
  194. ASSERT_EQ(buf[0], c[0]);
  195. ASSERT_EQ(buf[1], c[1]);
  196. ASSERT_EQ(buf[2], c[2]);
  197. ASSERT_EQ(buf[3], a[3]);
  198. ASSERT_EQ(Piece::npos, TypeParam::npos);
  199. ASSERT_EQ(a.find(b), 0U);
  200. ASSERT_EQ(a.find(b, 1), Piece::npos);
  201. ASSERT_EQ(a.find(c), 23U);
  202. ASSERT_EQ(a.find(c, 9), 23U);
  203. ASSERT_EQ(a.find(c, Piece::npos), Piece::npos);
  204. ASSERT_EQ(b.find(c), Piece::npos);
  205. ASSERT_EQ(b.find(c, Piece::npos), Piece::npos);
  206. ASSERT_EQ(a.find(d), 0U);
  207. ASSERT_EQ(a.find(e), 0U);
  208. ASSERT_EQ(a.find(d, 12), 12U);
  209. ASSERT_EQ(a.find(e, 17), 17U);
  210. TypeParam not_found(TestFixture::as_string("xx not found bb"));
  211. Piece g(not_found);
  212. ASSERT_EQ(a.find(g), Piece::npos);
  213. // empty string nonsense
  214. ASSERT_EQ(d.find(b), Piece::npos);
  215. ASSERT_EQ(e.find(b), Piece::npos);
  216. ASSERT_EQ(d.find(b, 4), Piece::npos);
  217. ASSERT_EQ(e.find(b, 7), Piece::npos);
  218. size_t empty_search_pos = TypeParam().find(TypeParam());
  219. ASSERT_EQ(d.find(d), empty_search_pos);
  220. ASSERT_EQ(d.find(e), empty_search_pos);
  221. ASSERT_EQ(e.find(d), empty_search_pos);
  222. ASSERT_EQ(e.find(e), empty_search_pos);
  223. ASSERT_EQ(d.find(d, 4), std::string().find(std::string(), 4));
  224. ASSERT_EQ(d.find(e, 4), std::string().find(std::string(), 4));
  225. ASSERT_EQ(e.find(d, 4), std::string().find(std::string(), 4));
  226. ASSERT_EQ(e.find(e, 4), std::string().find(std::string(), 4));
  227. ASSERT_EQ(a.find('a'), 0U);
  228. ASSERT_EQ(a.find('c'), 2U);
  229. ASSERT_EQ(a.find('z'), 25U);
  230. ASSERT_EQ(a.find('$'), Piece::npos);
  231. ASSERT_EQ(a.find('\0'), Piece::npos);
  232. ASSERT_EQ(f.find('\0'), 3U);
  233. ASSERT_EQ(f.find('3'), 2U);
  234. ASSERT_EQ(f.find('5'), 5U);
  235. ASSERT_EQ(g.find('o'), 4U);
  236. ASSERT_EQ(g.find('o', 4), 4U);
  237. ASSERT_EQ(g.find('o', 5), 8U);
  238. ASSERT_EQ(a.find('b', 5), Piece::npos);
  239. // empty string nonsense
  240. ASSERT_EQ(d.find('\0'), Piece::npos);
  241. ASSERT_EQ(e.find('\0'), Piece::npos);
  242. ASSERT_EQ(d.find('\0', 4), Piece::npos);
  243. ASSERT_EQ(e.find('\0', 7), Piece::npos);
  244. ASSERT_EQ(d.find('x'), Piece::npos);
  245. ASSERT_EQ(e.find('x'), Piece::npos);
  246. ASSERT_EQ(d.find('x', 4), Piece::npos);
  247. ASSERT_EQ(e.find('x', 7), Piece::npos);
  248. ASSERT_EQ(a.rfind(b), 0U);
  249. ASSERT_EQ(a.rfind(b, 1), 0U);
  250. ASSERT_EQ(a.rfind(c), 23U);
  251. ASSERT_EQ(a.rfind(c, 22U), Piece::npos);
  252. ASSERT_EQ(a.rfind(c, 1U), Piece::npos);
  253. ASSERT_EQ(a.rfind(c, 0U), Piece::npos);
  254. ASSERT_EQ(b.rfind(c), Piece::npos);
  255. ASSERT_EQ(b.rfind(c, 0U), Piece::npos);
  256. ASSERT_EQ(a.rfind(d), static_cast<size_t>(a.as_string().rfind(TypeParam())));
  257. ASSERT_EQ(a.rfind(e), a.as_string().rfind(TypeParam()));
  258. ASSERT_EQ(a.rfind(d, 12), 12U);
  259. ASSERT_EQ(a.rfind(e, 17), 17U);
  260. ASSERT_EQ(a.rfind(g), Piece::npos);
  261. ASSERT_EQ(d.rfind(b), Piece::npos);
  262. ASSERT_EQ(e.rfind(b), Piece::npos);
  263. ASSERT_EQ(d.rfind(b, 4), Piece::npos);
  264. ASSERT_EQ(e.rfind(b, 7), Piece::npos);
  265. // empty string nonsense
  266. ASSERT_EQ(d.rfind(d, 4), std::string().rfind(std::string()));
  267. ASSERT_EQ(e.rfind(d, 7), std::string().rfind(std::string()));
  268. ASSERT_EQ(d.rfind(e, 4), std::string().rfind(std::string()));
  269. ASSERT_EQ(e.rfind(e, 7), std::string().rfind(std::string()));
  270. ASSERT_EQ(d.rfind(d), std::string().rfind(std::string()));
  271. ASSERT_EQ(e.rfind(d), std::string().rfind(std::string()));
  272. ASSERT_EQ(d.rfind(e), std::string().rfind(std::string()));
  273. ASSERT_EQ(e.rfind(e), std::string().rfind(std::string()));
  274. ASSERT_EQ(g.rfind('o'), 8U);
  275. ASSERT_EQ(g.rfind('q'), Piece::npos);
  276. ASSERT_EQ(g.rfind('o', 8), 8U);
  277. ASSERT_EQ(g.rfind('o', 7), 4U);
  278. ASSERT_EQ(g.rfind('o', 3), Piece::npos);
  279. ASSERT_EQ(f.rfind('\0'), 3U);
  280. ASSERT_EQ(f.rfind('\0', 12), 3U);
  281. ASSERT_EQ(f.rfind('3'), 2U);
  282. ASSERT_EQ(f.rfind('5'), 5U);
  283. // empty string nonsense
  284. ASSERT_EQ(d.rfind('o'), Piece::npos);
  285. ASSERT_EQ(e.rfind('o'), Piece::npos);
  286. ASSERT_EQ(d.rfind('o', 4), Piece::npos);
  287. ASSERT_EQ(e.rfind('o', 7), Piece::npos);
  288. TypeParam one_two_three_four(TestFixture::as_string("one,two:three;four"));
  289. TypeParam comma_colon(TestFixture::as_string(",:"));
  290. ASSERT_EQ(3U, Piece(one_two_three_four).find_first_of(comma_colon));
  291. ASSERT_EQ(a.find_first_of(b), 0U);
  292. ASSERT_EQ(a.find_first_of(b, 0), 0U);
  293. ASSERT_EQ(a.find_first_of(b, 1), 1U);
  294. ASSERT_EQ(a.find_first_of(b, 2), 2U);
  295. ASSERT_EQ(a.find_first_of(b, 3), Piece::npos);
  296. ASSERT_EQ(a.find_first_of(c), 23U);
  297. ASSERT_EQ(a.find_first_of(c, 23), 23U);
  298. ASSERT_EQ(a.find_first_of(c, 24), 24U);
  299. ASSERT_EQ(a.find_first_of(c, 25), 25U);
  300. ASSERT_EQ(a.find_first_of(c, 26), Piece::npos);
  301. ASSERT_EQ(g.find_first_of(b), 13U);
  302. ASSERT_EQ(g.find_first_of(c), 0U);
  303. ASSERT_EQ(a.find_first_of(f), Piece::npos);
  304. ASSERT_EQ(f.find_first_of(a), Piece::npos);
  305. // empty string nonsense
  306. ASSERT_EQ(a.find_first_of(d), Piece::npos);
  307. ASSERT_EQ(a.find_first_of(e), Piece::npos);
  308. ASSERT_EQ(d.find_first_of(b), Piece::npos);
  309. ASSERT_EQ(e.find_first_of(b), Piece::npos);
  310. ASSERT_EQ(d.find_first_of(d), Piece::npos);
  311. ASSERT_EQ(e.find_first_of(d), Piece::npos);
  312. ASSERT_EQ(d.find_first_of(e), Piece::npos);
  313. ASSERT_EQ(e.find_first_of(e), Piece::npos);
  314. ASSERT_EQ(a.find_first_not_of(b), 3U);
  315. ASSERT_EQ(a.find_first_not_of(c), 0U);
  316. ASSERT_EQ(b.find_first_not_of(a), Piece::npos);
  317. ASSERT_EQ(c.find_first_not_of(a), Piece::npos);
  318. ASSERT_EQ(f.find_first_not_of(a), 0U);
  319. ASSERT_EQ(a.find_first_not_of(f), 0U);
  320. ASSERT_EQ(a.find_first_not_of(d), 0U);
  321. ASSERT_EQ(a.find_first_not_of(e), 0U);
  322. // empty string nonsense
  323. ASSERT_EQ(d.find_first_not_of(a), Piece::npos);
  324. ASSERT_EQ(e.find_first_not_of(a), Piece::npos);
  325. ASSERT_EQ(d.find_first_not_of(d), Piece::npos);
  326. ASSERT_EQ(e.find_first_not_of(d), Piece::npos);
  327. ASSERT_EQ(d.find_first_not_of(e), Piece::npos);
  328. ASSERT_EQ(e.find_first_not_of(e), Piece::npos);
  329. TypeParam equals(TestFixture::as_string("===="));
  330. Piece h(equals);
  331. ASSERT_EQ(h.find_first_not_of('='), Piece::npos);
  332. ASSERT_EQ(h.find_first_not_of('=', 3), Piece::npos);
  333. ASSERT_EQ(h.find_first_not_of('\0'), 0U);
  334. ASSERT_EQ(g.find_first_not_of('x'), 2U);
  335. ASSERT_EQ(f.find_first_not_of('\0'), 0U);
  336. ASSERT_EQ(f.find_first_not_of('\0', 3), 4U);
  337. ASSERT_EQ(f.find_first_not_of('\0', 2), 2U);
  338. // empty string nonsense
  339. ASSERT_EQ(d.find_first_not_of('x'), Piece::npos);
  340. ASSERT_EQ(e.find_first_not_of('x'), Piece::npos);
  341. ASSERT_EQ(d.find_first_not_of('\0'), Piece::npos);
  342. ASSERT_EQ(e.find_first_not_of('\0'), Piece::npos);
  343. // Piece g("xx not found bb");
  344. TypeParam fifty_six(TestFixture::as_string("56"));
  345. Piece i(fifty_six);
  346. ASSERT_EQ(h.find_last_of(a), Piece::npos);
  347. ASSERT_EQ(g.find_last_of(a), g.size()-1);
  348. ASSERT_EQ(a.find_last_of(b), 2U);
  349. ASSERT_EQ(a.find_last_of(c), a.size()-1);
  350. ASSERT_EQ(f.find_last_of(i), 6U);
  351. ASSERT_EQ(a.find_last_of('a'), 0U);
  352. ASSERT_EQ(a.find_last_of('b'), 1U);
  353. ASSERT_EQ(a.find_last_of('z'), 25U);
  354. ASSERT_EQ(a.find_last_of('a', 5), 0U);
  355. ASSERT_EQ(a.find_last_of('b', 5), 1U);
  356. ASSERT_EQ(a.find_last_of('b', 0), Piece::npos);
  357. ASSERT_EQ(a.find_last_of('z', 25), 25U);
  358. ASSERT_EQ(a.find_last_of('z', 24), Piece::npos);
  359. ASSERT_EQ(f.find_last_of(i, 5), 5U);
  360. ASSERT_EQ(f.find_last_of(i, 6), 6U);
  361. ASSERT_EQ(f.find_last_of(a, 4), Piece::npos);
  362. // empty string nonsense
  363. ASSERT_EQ(f.find_last_of(d), Piece::npos);
  364. ASSERT_EQ(f.find_last_of(e), Piece::npos);
  365. ASSERT_EQ(f.find_last_of(d, 4), Piece::npos);
  366. ASSERT_EQ(f.find_last_of(e, 4), Piece::npos);
  367. ASSERT_EQ(d.find_last_of(d), Piece::npos);
  368. ASSERT_EQ(d.find_last_of(e), Piece::npos);
  369. ASSERT_EQ(e.find_last_of(d), Piece::npos);
  370. ASSERT_EQ(e.find_last_of(e), Piece::npos);
  371. ASSERT_EQ(d.find_last_of(f), Piece::npos);
  372. ASSERT_EQ(e.find_last_of(f), Piece::npos);
  373. ASSERT_EQ(d.find_last_of(d, 4), Piece::npos);
  374. ASSERT_EQ(d.find_last_of(e, 4), Piece::npos);
  375. ASSERT_EQ(e.find_last_of(d, 4), Piece::npos);
  376. ASSERT_EQ(e.find_last_of(e, 4), Piece::npos);
  377. ASSERT_EQ(d.find_last_of(f, 4), Piece::npos);
  378. ASSERT_EQ(e.find_last_of(f, 4), Piece::npos);
  379. ASSERT_EQ(a.find_last_not_of(b), a.size()-1);
  380. ASSERT_EQ(a.find_last_not_of(c), 22U);
  381. ASSERT_EQ(b.find_last_not_of(a), Piece::npos);
  382. ASSERT_EQ(b.find_last_not_of(b), Piece::npos);
  383. ASSERT_EQ(f.find_last_not_of(i), 4U);
  384. ASSERT_EQ(a.find_last_not_of(c, 24), 22U);
  385. ASSERT_EQ(a.find_last_not_of(b, 3), 3U);
  386. ASSERT_EQ(a.find_last_not_of(b, 2), Piece::npos);
  387. // empty string nonsense
  388. ASSERT_EQ(f.find_last_not_of(d), f.size()-1);
  389. ASSERT_EQ(f.find_last_not_of(e), f.size()-1);
  390. ASSERT_EQ(f.find_last_not_of(d, 4), 4U);
  391. ASSERT_EQ(f.find_last_not_of(e, 4), 4U);
  392. ASSERT_EQ(d.find_last_not_of(d), Piece::npos);
  393. ASSERT_EQ(d.find_last_not_of(e), Piece::npos);
  394. ASSERT_EQ(e.find_last_not_of(d), Piece::npos);
  395. ASSERT_EQ(e.find_last_not_of(e), Piece::npos);
  396. ASSERT_EQ(d.find_last_not_of(f), Piece::npos);
  397. ASSERT_EQ(e.find_last_not_of(f), Piece::npos);
  398. ASSERT_EQ(d.find_last_not_of(d, 4), Piece::npos);
  399. ASSERT_EQ(d.find_last_not_of(e, 4), Piece::npos);
  400. ASSERT_EQ(e.find_last_not_of(d, 4), Piece::npos);
  401. ASSERT_EQ(e.find_last_not_of(e, 4), Piece::npos);
  402. ASSERT_EQ(d.find_last_not_of(f, 4), Piece::npos);
  403. ASSERT_EQ(e.find_last_not_of(f, 4), Piece::npos);
  404. ASSERT_EQ(h.find_last_not_of('x'), h.size() - 1);
  405. ASSERT_EQ(h.find_last_not_of('='), Piece::npos);
  406. ASSERT_EQ(b.find_last_not_of('c'), 1U);
  407. ASSERT_EQ(h.find_last_not_of('x', 2), 2U);
  408. ASSERT_EQ(h.find_last_not_of('=', 2), Piece::npos);
  409. ASSERT_EQ(b.find_last_not_of('b', 1), 0U);
  410. // empty string nonsense
  411. ASSERT_EQ(d.find_last_not_of('x'), Piece::npos);
  412. ASSERT_EQ(e.find_last_not_of('x'), Piece::npos);
  413. ASSERT_EQ(d.find_last_not_of('\0'), Piece::npos);
  414. ASSERT_EQ(e.find_last_not_of('\0'), Piece::npos);
  415. ASSERT_EQ(a.substr(0, 3), b);
  416. ASSERT_EQ(a.substr(23), c);
  417. ASSERT_EQ(a.substr(23, 3), c);
  418. ASSERT_EQ(a.substr(23, 99), c);
  419. ASSERT_EQ(a.substr(0), a);
  420. ASSERT_EQ(a.substr(3, 2), TestFixture::as_string("de"));
  421. // empty string nonsense
  422. ASSERT_EQ(a.substr(99, 2), e);
  423. ASSERT_EQ(d.substr(99), e);
  424. ASSERT_EQ(d.substr(0, 99), e);
  425. ASSERT_EQ(d.substr(99, 99), e);
  426. }
  427. TYPED_TEST(CommonStringPieceTest, CheckCustom) {
  428. TypeParam foobar(TestFixture::as_string("foobar"));
  429. BasicStringPiece<TypeParam> a(foobar);
  430. TypeParam s1(TestFixture::as_string("123"));
  431. s1 += static_cast<typename TypeParam::value_type>('\0');
  432. s1 += TestFixture::as_string("456");
  433. BasicStringPiece<TypeParam> b(s1);
  434. BasicStringPiece<TypeParam> e;
  435. TypeParam s2;
  436. // remove_prefix
  437. BasicStringPiece<TypeParam> c(a);
  438. c.remove_prefix(3);
  439. ASSERT_EQ(c, TestFixture::as_string("bar"));
  440. c = a;
  441. c.remove_prefix(0);
  442. ASSERT_EQ(c, a);
  443. c.remove_prefix(c.size());
  444. ASSERT_EQ(c, e);
  445. // remove_suffix
  446. c = a;
  447. c.remove_suffix(3);
  448. ASSERT_EQ(c, TestFixture::as_string("foo"));
  449. c = a;
  450. c.remove_suffix(0);
  451. ASSERT_EQ(c, a);
  452. c.remove_suffix(c.size());
  453. ASSERT_EQ(c, e);
  454. // set
  455. c.set(foobar.c_str());
  456. ASSERT_EQ(c, a);
  457. c.set(foobar.c_str(), 6);
  458. ASSERT_EQ(c, a);
  459. c.set(foobar.c_str(), 0);
  460. ASSERT_EQ(c, e);
  461. c.set(foobar.c_str(), 7); // Note, has an embedded NULL
  462. ASSERT_NE(c, a);
  463. // as_string
  464. TypeParam s3(a.as_string().c_str(), 7); // Note, has an embedded NULL
  465. ASSERT_TRUE(c == s3);
  466. TypeParam s4(e.as_string());
  467. ASSERT_TRUE(s4.empty());
  468. }
  469. TEST(StringPieceTest, CheckCustom) {
  470. StringPiece a("foobar");
  471. std::string s1("123");
  472. s1 += '\0';
  473. s1 += "456";
  474. StringPiece b(s1);
  475. StringPiece e;
  476. std::string s2;
  477. // CopyToString
  478. a.CopyToString(&s2);
  479. ASSERT_EQ(s2.size(), 6U);
  480. ASSERT_EQ(s2, "foobar");
  481. b.CopyToString(&s2);
  482. ASSERT_EQ(s2.size(), 7U);
  483. ASSERT_EQ(s1, s2);
  484. e.CopyToString(&s2);
  485. ASSERT_TRUE(s2.empty());
  486. // AppendToString
  487. s2.erase();
  488. a.AppendToString(&s2);
  489. ASSERT_EQ(s2.size(), 6U);
  490. ASSERT_EQ(s2, "foobar");
  491. a.AppendToString(&s2);
  492. ASSERT_EQ(s2.size(), 12U);
  493. ASSERT_EQ(s2, "foobarfoobar");
  494. // starts_with
  495. ASSERT_TRUE(a.starts_with(a));
  496. ASSERT_TRUE(a.starts_with("foo"));
  497. ASSERT_TRUE(a.starts_with(e));
  498. ASSERT_TRUE(b.starts_with(s1));
  499. ASSERT_TRUE(b.starts_with(b));
  500. ASSERT_TRUE(b.starts_with(e));
  501. ASSERT_TRUE(e.starts_with(""));
  502. ASSERT_TRUE(!a.starts_with(b));
  503. ASSERT_TRUE(!b.starts_with(a));
  504. ASSERT_TRUE(!e.starts_with(a));
  505. // ends with
  506. ASSERT_TRUE(a.ends_with(a));
  507. ASSERT_TRUE(a.ends_with("bar"));
  508. ASSERT_TRUE(a.ends_with(e));
  509. ASSERT_TRUE(b.ends_with(s1));
  510. ASSERT_TRUE(b.ends_with(b));
  511. ASSERT_TRUE(b.ends_with(e));
  512. ASSERT_TRUE(e.ends_with(""));
  513. ASSERT_TRUE(!a.ends_with(b));
  514. ASSERT_TRUE(!b.ends_with(a));
  515. ASSERT_TRUE(!e.ends_with(a));
  516. StringPiece c;
  517. c.set("foobar", 6);
  518. ASSERT_EQ(c, a);
  519. c.set("foobar", 0);
  520. ASSERT_EQ(c, e);
  521. c.set("foobar", 7);
  522. ASSERT_NE(c, a);
  523. }
  524. TYPED_TEST(CommonStringPieceTest, CheckNULL) {
  525. // we used to crash here, but now we don't.
  526. BasicStringPiece<TypeParam> s(NULL);
  527. ASSERT_EQ(s.data(), (const typename TypeParam::value_type*)NULL);
  528. ASSERT_EQ(s.size(), 0U);
  529. s.set(NULL);
  530. ASSERT_EQ(s.data(), (const typename TypeParam::value_type*)NULL);
  531. ASSERT_EQ(s.size(), 0U);
  532. TypeParam str = s.as_string();
  533. ASSERT_EQ(str.length(), 0U);
  534. ASSERT_EQ(str, TypeParam());
  535. }
  536. TYPED_TEST(CommonStringPieceTest, CheckComparisons2) {
  537. TypeParam alphabet(TestFixture::as_string("abcdefghijklmnopqrstuvwxyz"));
  538. TypeParam alphabet_z(TestFixture::as_string("abcdefghijklmnopqrstuvwxyzz"));
  539. TypeParam alphabet_y(TestFixture::as_string("abcdefghijklmnopqrstuvwxyy"));
  540. BasicStringPiece<TypeParam> abc(alphabet);
  541. // check comparison operations on strings longer than 4 bytes.
  542. ASSERT_TRUE(abc == BasicStringPiece<TypeParam>(alphabet));
  543. ASSERT_TRUE(abc.compare(BasicStringPiece<TypeParam>(alphabet)) == 0);
  544. ASSERT_TRUE(abc < BasicStringPiece<TypeParam>(alphabet_z));
  545. ASSERT_TRUE(abc.compare(BasicStringPiece<TypeParam>(alphabet_z)) < 0);
  546. ASSERT_TRUE(abc > BasicStringPiece<TypeParam>(alphabet_y));
  547. ASSERT_TRUE(abc.compare(BasicStringPiece<TypeParam>(alphabet_y)) > 0);
  548. }
  549. // Test operations only supported by std::string version.
  550. TEST(StringPieceTest, CheckComparisons2) {
  551. StringPiece abc("abcdefghijklmnopqrstuvwxyz");
  552. // starts_with
  553. ASSERT_TRUE(abc.starts_with(abc));
  554. ASSERT_TRUE(abc.starts_with("abcdefghijklm"));
  555. ASSERT_TRUE(!abc.starts_with("abcdefguvwxyz"));
  556. // ends_with
  557. ASSERT_TRUE(abc.ends_with(abc));
  558. ASSERT_TRUE(!abc.ends_with("abcdefguvwxyz"));
  559. ASSERT_TRUE(abc.ends_with("nopqrstuvwxyz"));
  560. }
  561. TYPED_TEST(CommonStringPieceTest, StringCompareNotAmbiguous) {
  562. ASSERT_TRUE(TestFixture::as_string("hello").c_str() ==
  563. TestFixture::as_string("hello"));
  564. ASSERT_TRUE(TestFixture::as_string("hello").c_str() <
  565. TestFixture::as_string("world"));
  566. }
  567. TYPED_TEST(CommonStringPieceTest, HeterogenousStringPieceEquals) {
  568. TypeParam hello(TestFixture::as_string("hello"));
  569. ASSERT_TRUE(BasicStringPiece<TypeParam>(hello) == hello);
  570. ASSERT_TRUE(hello.c_str() == BasicStringPiece<TypeParam>(hello));
  571. }
  572. // string16-specific stuff
  573. TEST(StringPiece16Test, CheckSTL) {
  574. // Check some non-ascii characters.
  575. string16 fifth(ASCIIToUTF16("123"));
  576. fifth.push_back(0x0000);
  577. fifth.push_back(0xd8c5);
  578. fifth.push_back(0xdffe);
  579. StringPiece16 f(fifth);
  580. ASSERT_EQ(f[3], '\0');
  581. ASSERT_EQ(f[5], static_cast<char16>(0xdffe));
  582. ASSERT_EQ(f.size(), 6U);
  583. }
  584. TEST(StringPiece16Test, CheckConversion) {
  585. // Make sure that we can convert from UTF8 to UTF16 and back. We use a two
  586. // byte character (G clef) to test this.
  587. ASSERT_EQ(
  588. UTF16ToUTF8(
  589. StringPiece16(UTF8ToUTF16("\xf0\x9d\x84\x9e")).as_string()),
  590. "\xf0\x9d\x84\x9e");
  591. }
  592. TYPED_TEST(CommonStringPieceTest, CheckConstructors) {
  593. TypeParam str(TestFixture::as_string("hello world"));
  594. TypeParam empty;
  595. ASSERT_TRUE(str == BasicStringPiece<TypeParam>(str));
  596. ASSERT_TRUE(str == BasicStringPiece<TypeParam>(str.c_str()));
  597. ASSERT_TRUE(TestFixture::as_string("hello") ==
  598. BasicStringPiece<TypeParam>(str.c_str(), 5));
  599. ASSERT_TRUE(empty == BasicStringPiece<TypeParam>(str.c_str(),
  600. static_cast<typename BasicStringPiece<TypeParam>::size_type>(0)));
  601. ASSERT_TRUE(empty == BasicStringPiece<TypeParam>(NULL));
  602. ASSERT_TRUE(empty == BasicStringPiece<TypeParam>(NULL,
  603. static_cast<typename BasicStringPiece<TypeParam>::size_type>(0)));
  604. ASSERT_TRUE(empty == BasicStringPiece<TypeParam>());
  605. ASSERT_TRUE(str == BasicStringPiece<TypeParam>(str.begin(), str.end()));
  606. ASSERT_TRUE(empty == BasicStringPiece<TypeParam>(str.begin(), str.begin()));
  607. ASSERT_TRUE(empty == BasicStringPiece<TypeParam>(empty));
  608. ASSERT_TRUE(empty == BasicStringPiece<TypeParam>(empty.begin(), empty.end()));
  609. }
  610. } // namespace butil