binary.cpp 3.1 KB

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