binary.cpp 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. #include "yaml-cpp/binary.h"
  2. #include <cctype>
  3. namespace YAML {
  4. static const char encoding[] =
  5. "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
  6. std::string EncodeBase64(const unsigned char *data, std::size_t size) {
  7. const char PAD = '=';
  8. std::string ret;
  9. ret.resize(4 * size / 3 + 3);
  10. char *out = &ret[0];
  11. std::size_t chunks = size / 3;
  12. std::size_t remainder = size % 3;
  13. for (std::size_t i = 0; i < chunks; i++, data += 3) {
  14. *out++ = encoding[data[0] >> 2];
  15. *out++ = encoding[((data[0] & 0x3) << 4) | (data[1] >> 4)];
  16. *out++ = encoding[((data[1] & 0xf) << 2) | (data[2] >> 6)];
  17. *out++ = encoding[data[2] & 0x3f];
  18. }
  19. switch (remainder) {
  20. case 0:
  21. break;
  22. case 1:
  23. *out++ = encoding[data[0] >> 2];
  24. *out++ = encoding[((data[0] & 0x3) << 4)];
  25. *out++ = PAD;
  26. *out++ = PAD;
  27. break;
  28. case 2:
  29. *out++ = encoding[data[0] >> 2];
  30. *out++ = encoding[((data[0] & 0x3) << 4) | (data[1] >> 4)];
  31. *out++ = encoding[((data[1] & 0xf) << 2)];
  32. *out++ = PAD;
  33. break;
  34. }
  35. ret.resize(out - &ret[0]);
  36. return ret;
  37. }
  38. static const unsigned char decoding[] = {
  39. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  40. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  41. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 62, 255,
  42. 255, 255, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 255, 255,
  43. 255, 0, 255, 255, 255, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,
  44. 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24,
  45. 25, 255, 255, 255, 255, 255, 255, 26, 27, 28, 29, 30, 31, 32, 33,
  46. 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48,
  47. 49, 50, 51, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  48. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  49. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  50. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  51. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  52. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  53. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  54. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  55. 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
  56. 255,
  57. };
  58. std::vector<unsigned char> DecodeBase64(const std::string &input) {
  59. using ret_type = std::vector<unsigned char>;
  60. if (input.empty())
  61. return ret_type();
  62. ret_type ret(3 * input.size() / 4 + 1);
  63. unsigned char *out = &ret[0];
  64. unsigned value = 0;
  65. for (std::size_t i = 0, cnt = 0; i < input.size(); i++) {
  66. if (std::isspace(static_cast<unsigned char>(input[i]))) {
  67. // skip newlines
  68. continue;
  69. }
  70. unsigned char d = decoding[static_cast<unsigned char>(input[i])];
  71. if (d == 255)
  72. return ret_type();
  73. value = (value << 6) | d;
  74. if (cnt % 4 == 3) {
  75. *out++ = value >> 16;
  76. if (i > 0 && input[i - 1] != '=')
  77. *out++ = value >> 8;
  78. if (input[i] != '=')
  79. *out++ = value;
  80. }
  81. ++cnt;
  82. }
  83. ret.resize(out - &ret[0]);
  84. return ret;
  85. }
  86. } // namespace YAML