base64.hpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  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 _BASE64_HPP_
  27. #define _BASE64_HPP_
  28. #include <string>
  29. namespace websocketpp {
  30. static std::string const base64_chars =
  31. "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  32. "abcdefghijklmnopqrstuvwxyz"
  33. "0123456789+/";
  34. /// Test whether a character is a valid base64 character
  35. /**
  36. * @param c The character to test
  37. * @return true if c is a valid base64 character
  38. */
  39. static inline bool is_base64(unsigned char c) {
  40. return (c == 43 || // +
  41. (c >= 47 && c <= 57) || // /-9
  42. (c >= 65 && c <= 90) || // A-Z
  43. (c >= 97 && c <= 122)); // a-z
  44. }
  45. /// Encode a char buffer into a base64 string
  46. /**
  47. * @param input The input data
  48. * @param len The length of input in bytes
  49. * @return A base64 encoded string representing input
  50. */
  51. inline std::string base64_encode(unsigned char const * input, size_t len) {
  52. std::string ret;
  53. int i = 0;
  54. int j = 0;
  55. unsigned char char_array_3[3];
  56. unsigned char char_array_4[4];
  57. while (len--) {
  58. char_array_3[i++] = *(input++);
  59. if (i == 3) {
  60. char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;
  61. char_array_4[1] = ((char_array_3[0] & 0x03) << 4) +
  62. ((char_array_3[1] & 0xf0) >> 4);
  63. char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) +
  64. ((char_array_3[2] & 0xc0) >> 6);
  65. char_array_4[3] = char_array_3[2] & 0x3f;
  66. for(i = 0; (i <4) ; i++) {
  67. ret += base64_chars[char_array_4[i]];
  68. }
  69. i = 0;
  70. }
  71. }
  72. if (i) {
  73. for(j = i; j < 3; j++) {
  74. char_array_3[j] = '\0';
  75. }
  76. char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;
  77. char_array_4[1] = ((char_array_3[0] & 0x03) << 4) +
  78. ((char_array_3[1] & 0xf0) >> 4);
  79. char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) +
  80. ((char_array_3[2] & 0xc0) >> 6);
  81. char_array_4[3] = char_array_3[2] & 0x3f;
  82. for (j = 0; (j < i + 1); j++) {
  83. ret += base64_chars[char_array_4[j]];
  84. }
  85. while((i++ < 3)) {
  86. ret += '=';
  87. }
  88. }
  89. return ret;
  90. }
  91. /// Encode a string into a base64 string
  92. /**
  93. * @param input The input data
  94. * @return A base64 encoded string representing input
  95. */
  96. inline std::string base64_encode(std::string const & input) {
  97. return base64_encode(
  98. reinterpret_cast<const unsigned char *>(input.data()),
  99. input.size()
  100. );
  101. }
  102. /// Decode a base64 encoded string into a string of raw bytes
  103. /**
  104. * @param input The base64 encoded input data
  105. * @return A string representing the decoded raw bytes
  106. */
  107. inline std::string base64_decode(std::string const & input) {
  108. size_t in_len = input.size();
  109. int i = 0;
  110. int j = 0;
  111. int in_ = 0;
  112. unsigned char char_array_4[4], char_array_3[3];
  113. std::string ret;
  114. while (in_len-- && ( input[in_] != '=') && is_base64(input[in_])) {
  115. char_array_4[i++] = input[in_]; in_++;
  116. if (i ==4) {
  117. for (i = 0; i <4; i++) {
  118. char_array_4[i] = static_cast<unsigned char>(base64_chars.find(char_array_4[i]));
  119. }
  120. char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
  121. char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
  122. char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];
  123. for (i = 0; (i < 3); i++) {
  124. ret += char_array_3[i];
  125. }
  126. i = 0;
  127. }
  128. }
  129. if (i) {
  130. for (j = i; j <4; j++)
  131. char_array_4[j] = 0;
  132. for (j = 0; j <4; j++)
  133. char_array_4[j] = static_cast<unsigned char>(base64_chars.find(char_array_4[j]));
  134. char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
  135. char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
  136. char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];
  137. for (j = 0; (j < i - 1); j++) {
  138. ret += static_cast<std::string::value_type>(char_array_3[j]);
  139. }
  140. }
  141. return ret;
  142. }
  143. } // namespace websocketpp
  144. #endif // _BASE64_HPP_