base64.hpp 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. /*
  2. ******
  3. base64.hpp is a repackaging of the base64.cpp and base64.h files into a
  4. single header suitable for use as a header only library. This conversion was
  5. done by Peter Thorson (webmaster@zaphoyd.com) in 2012. All modifications to
  6. the code are redistributed under the same license as the original, which is
  7. listed below.
  8. ******
  9. base64.cpp and base64.h
  10. Copyright (C) 2004-2008 Ren Nyffenegger
  11. This source code is provided 'as-is', without any express or implied
  12. warranty. In no event will the author be held liable for any damages
  13. arising from the use of this software.
  14. Permission is granted to anyone to use this software for any purpose,
  15. including commercial applications, and to alter it and redistribute it
  16. freely, subject to the following restrictions:
  17. 1. The origin of this source code must not be misrepresented; you must not
  18. claim that you wrote the original source code. If you use this source code
  19. in a product, an acknowledgment in the product documentation would be
  20. appreciated but is not required.
  21. 2. Altered source versions must be plainly marked as such, and must not be
  22. misrepresented as being the original source code.
  23. 3. This notice may not be removed or altered from any source distribution.
  24. Ren Nyffenegger rene.nyffenegger@adp-gmbh.ch
  25. */
  26. #ifndef __ASIO2_BASE64_IMPL_HPP__
  27. #define __ASIO2_BASE64_IMPL_HPP__
  28. #include <string>
  29. #include <string_view>
  30. namespace asio2 {
  31. namespace detail
  32. {
  33. static std::string_view const base64_chars =
  34. "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  35. "abcdefghijklmnopqrstuvwxyz"
  36. "0123456789+/";
  37. }
  38. class base64
  39. {
  40. public:
  41. /// Test whether a character is a valid base64 character
  42. /**
  43. * @param c The character to test
  44. * @return true if c is a valid base64 character
  45. */
  46. inline bool is_base64(unsigned char c) noexcept {
  47. return (c == 43 || // +
  48. (c >= 47 && c <= 57) || // /-9
  49. (c >= 65 && c <= 90) || // A-Z
  50. (c >= 97 && c <= 122)); // a-z
  51. }
  52. /// Encode a char buffer into a base64 string
  53. /**
  54. * @param input The input data
  55. * @param len The length of input in bytes
  56. * @return A base64 encoded string representing input
  57. */
  58. inline std::string encode(unsigned char const * input, size_t len) {
  59. std::string ret;
  60. int i = 0;
  61. int j = 0;
  62. unsigned char char_array_3[3];
  63. unsigned char char_array_4[4];
  64. while (len--) {
  65. char_array_3[i++] = *(input++);
  66. if (i == 3) {
  67. char_array_4[0] = (unsigned char)((char_array_3[0] & 0xfc) >> 2);
  68. char_array_4[1] = (unsigned char)(((char_array_3[0] & 0x03) << 4) +
  69. ((char_array_3[1] & 0xf0) >> 4));
  70. char_array_4[2] = (unsigned char)(((char_array_3[1] & 0x0f) << 2) +
  71. ((char_array_3[2] & 0xc0) >> 6));
  72. char_array_4[3] = (unsigned char)(char_array_3[2] & 0x3f);
  73. for(i = 0; (i <4) ; i++) {
  74. ret += detail::base64_chars[char_array_4[i]];
  75. }
  76. i = 0;
  77. }
  78. }
  79. if (i) {
  80. for(j = i; j < 3; j++) {
  81. char_array_3[j] = '\0';
  82. }
  83. char_array_4[0] = (unsigned char)((char_array_3[0] & 0xfc) >> 2);
  84. char_array_4[1] = (unsigned char)(((char_array_3[0] & 0x03) << 4) +
  85. ((char_array_3[1] & 0xf0) >> 4));
  86. char_array_4[2] = (unsigned char)(((char_array_3[1] & 0x0f) << 2) +
  87. ((char_array_3[2] & 0xc0) >> 6));
  88. char_array_4[3] = (unsigned char)(char_array_3[2] & 0x3f);
  89. for (j = 0; (j < i + 1); j++) {
  90. ret += detail::base64_chars[char_array_4[j]];
  91. }
  92. while((i++ < 3)) {
  93. ret += '=';
  94. }
  95. }
  96. return ret;
  97. }
  98. /// Encode a string into a base64 string
  99. /**
  100. * @param input The input data
  101. * @return A base64 encoded string representing input
  102. */
  103. inline std::string encode(std::string_view const & input) {
  104. return encode(
  105. reinterpret_cast<const unsigned char *>(input.data()),
  106. input.size()
  107. );
  108. }
  109. /// Decode a base64 encoded string into a string of raw bytes
  110. /**
  111. * @param input The base64 encoded input data
  112. * @return A string representing the decoded raw bytes
  113. */
  114. inline std::string decode(std::string_view const & input) {
  115. size_t in_len = input.size();
  116. int i = 0;
  117. int j = 0;
  118. int in_ = 0;
  119. unsigned char char_array_4[4], char_array_3[3];
  120. std::string ret;
  121. while (in_len-- && ( input[in_] != '=') && is_base64(input[in_])) {
  122. char_array_4[i++] = input[in_]; in_++;
  123. if (i ==4) {
  124. for (i = 0; i <4; i++) {
  125. char_array_4[i] = static_cast<unsigned char>(detail::base64_chars.find(char_array_4[i]));
  126. }
  127. char_array_3[0] = (unsigned char)((char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4));
  128. char_array_3[1] = (unsigned char)(((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2));
  129. char_array_3[2] = (unsigned char)(((char_array_4[2] & 0x3) << 6) + char_array_4[3]);
  130. for (i = 0; (i < 3); i++) {
  131. ret += char_array_3[i];
  132. }
  133. i = 0;
  134. }
  135. }
  136. if (i) {
  137. for (j = i; j <4; j++)
  138. char_array_4[j] = 0;
  139. for (j = 0; j <4; j++)
  140. char_array_4[j] = static_cast<unsigned char>(detail::base64_chars.find(char_array_4[j]));
  141. char_array_3[0] = (unsigned char)((char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4));
  142. char_array_3[1] = (unsigned char)(((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2));
  143. char_array_3[2] = (unsigned char)(((char_array_4[2] & 0x3) << 6) + char_array_4[3]);
  144. for (j = 0; (j < i - 1); j++) {
  145. ret += static_cast<std::string::value_type>(char_array_3[j]);
  146. }
  147. }
  148. return ret;
  149. }
  150. };
  151. /// Test whether a character is a valid base64 character
  152. /**
  153. * @param c The character to test
  154. * @return true if c is a valid base64 character
  155. */
  156. inline bool is_base64(unsigned char c) noexcept
  157. {
  158. return base64().is_base64(c);
  159. }
  160. /// Encode a char buffer into a base64 string
  161. /**
  162. * @param input The input data
  163. * @param len The length of input in bytes
  164. * @return A base64 encoded string representing input
  165. */
  166. inline std::string base64_encode(unsigned char const * input, size_t len)
  167. {
  168. return base64().encode(input, len);
  169. }
  170. /// Encode a string into a base64 string
  171. /**
  172. * @param input The input data
  173. * @return A base64 encoded string representing input
  174. */
  175. inline std::string base64_encode(std::string_view const & input)
  176. {
  177. return base64().encode(input);
  178. }
  179. /// Decode a base64 encoded string into a string of raw bytes
  180. /**
  181. * @param input The base64 encoded input data
  182. * @return A string representing the decoded raw bytes
  183. */
  184. inline std::string base64_decode(std::string_view const & input)
  185. {
  186. return base64().decode(input);
  187. }
  188. } // namespace asio2
  189. #endif // __ASIO2_BASE64_IMPL_HPP__