base64.cpp 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // Name: tests/base64/base64.cpp
  3. // Purpose: wxBase64Encode/Decode unit test
  4. // Author: Charles Reimers
  5. // Created: 2007-06-22
  6. ///////////////////////////////////////////////////////////////////////////////
  7. // ----------------------------------------------------------------------------
  8. // headers
  9. // ----------------------------------------------------------------------------
  10. #include "testprec.h"
  11. #ifdef __BORLANDC__
  12. #pragma hdrstop
  13. #endif
  14. #ifndef WX_PRECOMP
  15. #include "wx/wx.h"
  16. #endif // WX_PRECOMP
  17. #if wxUSE_BASE64
  18. #include "wx/base64.h"
  19. static const char encoded0to255[] =
  20. "AAECAwQFBgcICQoLDA0ODxAREhMUFRYXGBkaGxwdHh8gISIj"
  21. "JCUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZH"
  22. "SElKS0xNTk9QUVJTVFVWV1hZWltcXV5fYGFiY2RlZmdoaWpr"
  23. "bG1ub3BxcnN0dXZ3eHl6e3x9fn+AgYKDhIWGh4iJiouMjY6P"
  24. "kJGSk5SVlpeYmZqbnJ2en6ChoqOkpaanqKmqq6ytrq+wsbKz"
  25. "tLW2t7i5uru8vb6/wMHCw8TFxsfIycrLzM3Oz9DR0tPU1dbX"
  26. "2Nna29zd3t/g4eLj5OXm5+jp6uvs7e7v8PHy8/T19vf4+fr7"
  27. "/P3+/w==";
  28. static void
  29. generatePatternedData(void* buff, size_t len, unsigned char startVal,
  30. unsigned char addVal, unsigned char multVal = 1,
  31. unsigned char xorMask = 0, unsigned char andMask = 255)
  32. {
  33. unsigned char *cbuff = (unsigned char *)buff;
  34. unsigned char curval = startVal;
  35. while(len--)
  36. {
  37. *(cbuff++) = curval;
  38. curval = (((curval + addVal) * multVal) ^ xorMask) & andMask;
  39. }
  40. }
  41. static void generateRandomData(void* buff, size_t len)
  42. {
  43. unsigned char *cbuff = (unsigned char *)buff;
  44. while(len--)
  45. {
  46. *(cbuff++) = (unsigned char)(((rand() * 255) / RAND_MAX));
  47. }
  48. }
  49. static void generateGibberish(void* buff, size_t len)
  50. {
  51. static const unsigned char cb64[] =
  52. "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
  53. unsigned char *cbuff = (unsigned char *)buff;
  54. while(len--)
  55. {
  56. *(cbuff++) = cb64[((rand() * 64) / RAND_MAX)];
  57. }
  58. }
  59. // --------------------------------------------------------------------------
  60. // test class
  61. // --------------------------------------------------------------------------
  62. class Base64TestCase : public CppUnit::TestCase
  63. {
  64. public:
  65. Base64TestCase() { }
  66. private:
  67. CPPUNIT_TEST_SUITE( Base64TestCase );
  68. CPPUNIT_TEST( EncodeDecodeEmpty );
  69. CPPUNIT_TEST( EncodeDecodeA );
  70. CPPUNIT_TEST( EncodeDecodeAB );
  71. CPPUNIT_TEST( EncodeDecodeABC );
  72. CPPUNIT_TEST( EncodeDecodeABCD );
  73. CPPUNIT_TEST( EncodeDecode0to255 );
  74. CPPUNIT_TEST( EncodeDecodePatternA );
  75. CPPUNIT_TEST( EncodeDecodePatternB );
  76. CPPUNIT_TEST( EncodeDecodePatternC );
  77. CPPUNIT_TEST( EncodeDecodeRandom );
  78. CPPUNIT_TEST( DecodeInvalid );
  79. CPPUNIT_TEST_SUITE_END();
  80. void EncodeDecodeEmpty();
  81. void EncodeDecodeA();
  82. void EncodeDecodeAB();
  83. void EncodeDecodeABC();
  84. void EncodeDecodeABCD();
  85. void EncodeDecode0to255();
  86. void EncodeDecodePatternA();
  87. void EncodeDecodePatternB();
  88. void EncodeDecodePatternC();
  89. void EncodeDecodeRandom();
  90. void DecodeInvalid();
  91. DECLARE_NO_COPY_CLASS(Base64TestCase)
  92. };
  93. // register in the unnamed registry so that these tests are run by default
  94. CPPUNIT_TEST_SUITE_REGISTRATION( Base64TestCase );
  95. // also include in its own registry so that these tests can be run alone
  96. CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( Base64TestCase, "Base64TestCase" );
  97. void Base64TestCase::EncodeDecodeEmpty()
  98. {
  99. char shouldBeEmpty[10];
  100. shouldBeEmpty[0] = '\0';
  101. size_t len = 10;
  102. CPPUNIT_ASSERT(wxBase64Encode(shouldBeEmpty, len, "", 0) != wxCONV_FAILED);
  103. CPPUNIT_ASSERT_EQUAL('\0', shouldBeEmpty[0]);
  104. CPPUNIT_ASSERT(wxBase64Decode(shouldBeEmpty, len, "") != wxCONV_FAILED);
  105. CPPUNIT_ASSERT_EQUAL('\0', shouldBeEmpty[0]);
  106. wxMemoryBuffer bufmt;
  107. wxString resultEmpty = wxBase64Encode(bufmt);
  108. CPPUNIT_ASSERT(resultEmpty.empty());
  109. bufmt = wxBase64Decode(resultEmpty);
  110. CPPUNIT_ASSERT_EQUAL(0, bufmt.GetDataLen());
  111. }
  112. void Base64TestCase::EncodeDecodeA()
  113. {
  114. const wxString str = wxBase64Encode("A", 1);
  115. CPPUNIT_ASSERT_EQUAL(wxString("QQ=="), str);
  116. wxMemoryBuffer buf = wxBase64Decode(str);
  117. CPPUNIT_ASSERT_EQUAL(1, buf.GetDataLen());
  118. CPPUNIT_ASSERT_EQUAL('A', *(char *)buf.GetData());
  119. char cbuf[10];
  120. memset(cbuf, (char)-1, sizeof(cbuf));
  121. CPPUNIT_ASSERT_EQUAL( 1, wxBase64Decode(cbuf, 1, str) );
  122. CPPUNIT_ASSERT_EQUAL( 'A', cbuf[0] );
  123. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[1] );
  124. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[2] );
  125. }
  126. void Base64TestCase::EncodeDecodeAB()
  127. {
  128. const wxString str = wxBase64Encode("AB", 2);
  129. CPPUNIT_ASSERT_EQUAL(wxString("QUI="), str);
  130. wxMemoryBuffer buf = wxBase64Decode(str);
  131. CPPUNIT_ASSERT_EQUAL(2, buf.GetDataLen());
  132. CPPUNIT_ASSERT_EQUAL('A', buf[0]);
  133. CPPUNIT_ASSERT_EQUAL('B', buf[1]);
  134. char cbuf[10];
  135. memset(cbuf, (char)-1, sizeof(cbuf));
  136. CPPUNIT_ASSERT_EQUAL( wxCONV_FAILED, wxBase64Decode(cbuf, 1, str) );
  137. CPPUNIT_ASSERT_EQUAL( 2, wxBase64Decode(cbuf, 2, str) );
  138. CPPUNIT_ASSERT_EQUAL( 'A', cbuf[0] );
  139. CPPUNIT_ASSERT_EQUAL( 'B', cbuf[1] );
  140. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[2] );
  141. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[3] );
  142. }
  143. void Base64TestCase::EncodeDecodeABC()
  144. {
  145. const wxString str = wxBase64Encode("ABC", 3);
  146. CPPUNIT_ASSERT_EQUAL(wxString("QUJD"), str);
  147. wxMemoryBuffer buf = wxBase64Decode(str);
  148. CPPUNIT_ASSERT_EQUAL(3, buf.GetDataLen());
  149. CPPUNIT_ASSERT_EQUAL('A', buf[0]);
  150. CPPUNIT_ASSERT_EQUAL('B', buf[1]);
  151. CPPUNIT_ASSERT_EQUAL('C', buf[2]);
  152. char cbuf[10];
  153. memset(cbuf, (char)-1, sizeof(cbuf));
  154. CPPUNIT_ASSERT_EQUAL( wxCONV_FAILED, wxBase64Decode(cbuf, 2, str) );
  155. CPPUNIT_ASSERT_EQUAL( 3, wxBase64Decode(cbuf, 3, str) );
  156. CPPUNIT_ASSERT_EQUAL( 'A', cbuf[0] );
  157. CPPUNIT_ASSERT_EQUAL( 'B', cbuf[1] );
  158. CPPUNIT_ASSERT_EQUAL( 'C', cbuf[2] );
  159. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[3] );
  160. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[4] );
  161. }
  162. void Base64TestCase::EncodeDecodeABCD()
  163. {
  164. const wxString str = wxBase64Encode("ABCD", 4);
  165. CPPUNIT_ASSERT_EQUAL(wxString("QUJDRA=="), str);
  166. wxMemoryBuffer buf = wxBase64Decode(str);
  167. CPPUNIT_ASSERT_EQUAL(4, buf.GetDataLen());
  168. CPPUNIT_ASSERT_EQUAL('A', buf[0]);
  169. CPPUNIT_ASSERT_EQUAL('B', buf[1]);
  170. CPPUNIT_ASSERT_EQUAL('C', buf[2]);
  171. CPPUNIT_ASSERT_EQUAL('D', buf[3]);
  172. char cbuf[10];
  173. memset(cbuf, (char)-1, sizeof(cbuf));
  174. CPPUNIT_ASSERT_EQUAL( wxCONV_FAILED, wxBase64Decode(cbuf, 3, str) );
  175. CPPUNIT_ASSERT_EQUAL( 4, wxBase64Decode(cbuf, 4, str) );
  176. CPPUNIT_ASSERT_EQUAL( 'A', cbuf[0] );
  177. CPPUNIT_ASSERT_EQUAL( 'B', cbuf[1] );
  178. CPPUNIT_ASSERT_EQUAL( 'C', cbuf[2] );
  179. CPPUNIT_ASSERT_EQUAL( 'D', cbuf[3] );
  180. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[4] );
  181. CPPUNIT_ASSERT_EQUAL( (char)-1, cbuf[5] );
  182. }
  183. void Base64TestCase::EncodeDecode0to255()
  184. {
  185. unsigned char buff[256];
  186. generatePatternedData(buff, 256, 0, 1);
  187. wxString str = wxBase64Encode(buff, 256);
  188. wxMemoryBuffer mbuff = wxBase64Decode(str);
  189. CPPUNIT_ASSERT(memcmp(mbuff.GetData(), buff, mbuff.GetDataLen()) == 0);
  190. mbuff = wxBase64Decode(encoded0to255);
  191. CPPUNIT_ASSERT(memcmp(mbuff.GetData(), buff, mbuff.GetDataLen()) == 0);
  192. }
  193. void Base64TestCase::EncodeDecodePatternA()
  194. {
  195. unsigned char buff[350];
  196. generatePatternedData(buff, 350, 24, 5, 3);
  197. wxString str = wxBase64Encode(buff, 350);
  198. wxMemoryBuffer mbuff = wxBase64Decode(str);
  199. CPPUNIT_ASSERT(memcmp(mbuff.GetData(), buff, mbuff.GetDataLen()) == 0);
  200. }
  201. void Base64TestCase::EncodeDecodePatternB()
  202. {
  203. unsigned char buff[350];
  204. generatePatternedData(buff, 350, 0, 1, 1, 0xAA);
  205. wxString str = wxBase64Encode(buff, 350);
  206. wxMemoryBuffer mbuff = wxBase64Decode(str);
  207. CPPUNIT_ASSERT(memcmp(mbuff.GetData(), buff, mbuff.GetDataLen()) == 0);
  208. }
  209. void Base64TestCase::EncodeDecodePatternC()
  210. {
  211. unsigned char buff[11];
  212. generatePatternedData(buff, 11, 1, 0, 2);
  213. wxString str = wxBase64Encode(buff, 11);
  214. wxMemoryBuffer mbuff = wxBase64Decode(str);
  215. CPPUNIT_ASSERT(memcmp(mbuff.GetData(), buff, mbuff.GetDataLen()) == 0);
  216. }
  217. void Base64TestCase::EncodeDecodeRandom()
  218. {
  219. size_t size = rand() * 3000 / RAND_MAX + 11;
  220. unsigned char *buff = new unsigned char[size];
  221. generateRandomData(buff, size);
  222. wxString str = wxBase64Encode(buff, size);
  223. wxMemoryBuffer mbuff = wxBase64Decode(str);
  224. CPPUNIT_ASSERT(memcmp(mbuff.GetData(), buff, mbuff.GetDataLen()) == 0);
  225. generateGibberish(buff, size);
  226. char *buff2 = new char[size];
  227. size_t realsize = size;
  228. CPPUNIT_ASSERT(wxBase64Decode(buff2, realsize, (char *)buff, size));
  229. CPPUNIT_ASSERT(wxBase64Encode(buff2, size, buff2, realsize));
  230. }
  231. void Base64TestCase::DecodeInvalid()
  232. {
  233. size_t rc, posErr;
  234. rc = wxBase64Decode(NULL, 0, "one two!", wxNO_LEN,
  235. wxBase64DecodeMode_Strict, &posErr);
  236. CPPUNIT_ASSERT_EQUAL( wxCONV_FAILED, rc);
  237. CPPUNIT_ASSERT_EQUAL( 3, posErr );
  238. rc = wxBase64Decode(NULL, 0, "one two!", wxNO_LEN,
  239. wxBase64DecodeMode_SkipWS, &posErr);
  240. CPPUNIT_ASSERT_EQUAL( wxCONV_FAILED, rc);
  241. CPPUNIT_ASSERT_EQUAL( 7, posErr );
  242. rc = wxBase64Decode(NULL, 0, "? QQ==", wxNO_LEN,
  243. wxBase64DecodeMode_SkipWS, &posErr);
  244. CPPUNIT_ASSERT_EQUAL( wxCONV_FAILED, rc);
  245. CPPUNIT_ASSERT_EQUAL( 0, posErr );
  246. posErr = (size_t)-1;
  247. rc = wxBase64Decode(NULL, 0, " QQ==", wxNO_LEN,
  248. wxBase64DecodeMode_SkipWS, &posErr);
  249. CPPUNIT_ASSERT_EQUAL( 1, rc );
  250. CPPUNIT_ASSERT_EQUAL( -1, posErr );
  251. rc = wxBase64Decode(NULL, 0, "? QQ==", wxNO_LEN,
  252. wxBase64DecodeMode_Relaxed, &posErr);
  253. CPPUNIT_ASSERT_EQUAL( 1, rc );
  254. CPPUNIT_ASSERT_EQUAL( -1, posErr );
  255. CPPUNIT_ASSERT( !wxBase64Decode("wxGetApp()").GetDataLen() );
  256. }
  257. #endif // wxUSE_BASE64