Base64.cpp 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. /*
  2. * Author: Patrick-Christopher Mattulat
  3. * Co-Author: Claude Sonnet 4.6 (LLM)
  4. * Company: Lynar Studios
  5. * E-Mail: webmaster@lynarstudios.com
  6. * Created: 2022-01-03
  7. * Changed: 2026-06-23
  8. *
  9. * */
  10. #include <bitset>
  11. #include <ls-std/encoding/Base64.hpp>
  12. using ls::standard::encoding::Base64;
  13. using std::string;
  14. using std::string_view;
  15. using std::unordered_map;
  16. using std::vector;
  17. Base64::Base64() = default;
  18. Base64::~Base64() noexcept = default;
  19. string Base64::encode(const string &_sequence)
  20. {
  21. string encodedString{};
  22. for (size_t index = 0; index < _sequence.size(); index += 3)
  23. {
  24. const string_view byteTriple = _getNextByteTriple(_sequence, index);
  25. encodedString += _encodeByteTriple(byteTriple);
  26. }
  27. return _applyEndingRule(encodedString, _sequence.size());
  28. }
  29. string Base64::decode(const string &_sequence)
  30. {
  31. string decodedString{};
  32. for (size_t index{}; index < _sequence.size(); index += 4)
  33. {
  34. const string_view quadruple = _getNextByteQuadruple(string_view{_sequence}, index);
  35. decodedString += _decodeByteQuadruple(quadruple);
  36. }
  37. return decodedString;
  38. }
  39. string Base64::_applyEndingRule(string _encodedString, const size_t _sequenceSize)
  40. {
  41. const size_t size = _encodedString.size();
  42. if (_sequenceSize % 3 == 1)
  43. {
  44. _encodedString[size - 2] = '=';
  45. _encodedString[size - 1] = '=';
  46. }
  47. if (_sequenceSize % 3 == 2)
  48. {
  49. _encodedString[size - 1] = '=';
  50. }
  51. return _encodedString;
  52. }
  53. string Base64::_decodeByteQuadruple(const string_view _quadruple)
  54. {
  55. string decodedText{};
  56. uint8_t shiftValue = 16;
  57. const uint32_t bitStorage = _toDecodingBitStorage(_quadruple);
  58. for (uint8_t index = 0; index < static_cast<uint8_t>(_quadruple.size()) - 1; index++)
  59. {
  60. const uint32_t bitMask = _generateBitMask(255, shiftValue);
  61. uint32_t bitSequence = _extractBitSequence(bitMask, bitStorage);
  62. bitSequence = bitSequence >> shiftValue;
  63. decodedText += static_cast<char>(bitSequence);
  64. shiftValue -= 8;
  65. }
  66. return decodedText;
  67. }
  68. string Base64::_encodeByteTriple(const string_view _byteTriple)
  69. {
  70. string encodedText{};
  71. const uint32_t bitStorage = _toEncodingBitStorage(_byteTriple);
  72. static vector<uint32_t> bitMaskStorage = {16515072, 258048, 4032, 63};
  73. static unordered_map<uint8_t, char> encodingMap = _getEncodingMap();
  74. uint8_t shiftValue = 18;
  75. for (uint8_t bitMaskIndex = 0; bitMaskIndex < 4; bitMaskIndex++)
  76. {
  77. uint32_t extractedBitSequence = _extractBitSequence(bitMaskStorage[bitMaskIndex], bitStorage);
  78. extractedBitSequence = extractedBitSequence >> shiftValue;
  79. encodedText += encodingMap[static_cast<uint8_t>(extractedBitSequence)];
  80. shiftValue -= 6;
  81. }
  82. return encodedText;
  83. }
  84. uint32_t Base64::_extractBitSequence(const uint32_t _bitMask, const uint32_t _bitStorage)
  85. {
  86. return _bitStorage & _bitMask;
  87. }
  88. uint32_t Base64::_generateBitMask(const uint32_t _maskValue, const uint8_t _shiftValue)
  89. {
  90. uint32_t bitmask = _maskValue << _shiftValue;
  91. if (_shiftValue == 0)
  92. {
  93. bitmask = _maskValue;
  94. }
  95. return bitmask;
  96. }
  97. unordered_map<char, uint8_t> Base64::_getDecodingMap()
  98. {
  99. static unordered_map<char, uint8_t> decodingMap = {{'A', 0}, {'B', 1}, {'C', 2}, {'D', 3}, {'E', 4}, {'F', 5}, {'G', 6}, {'H', 7}, {'I', 8}, {'J', 9}, {'K', 10}, {'L', 11}, {'M', 12}, {'N', 13}, {'O', 14}, {'P', 15}, {'Q', 16}, {'R', 17}, {'S', 18}, {'T', 19}, {'U', 20}, {'V', 21}, {'W', 22}, {'X', 23}, {'Y', 24}, {'Z', 25}, {'a', 26}, {'b', 27}, {'c', 28}, {'d', 29}, {'e', 30}, {'f', 31},
  100. {'g', 32}, {'h', 33}, {'i', 34}, {'j', 35}, {'k', 36}, {'l', 37}, {'m', 38}, {'n', 39}, {'o', 40}, {'p', 41}, {'q', 42}, {'r', 43}, {'s', 44}, {'t', 45}, {'u', 46}, {'v', 47}, {'w', 48}, {'x', 49}, {'y', 50}, {'z', 51}, {'0', 52}, {'1', 53}, {'2', 54}, {'3', 55}, {'4', 56}, {'5', 57}, {'6', 58}, {'7', 59}, {'8', 60}, {'9', 61}, {'+', 62}, {'/', 63}};
  101. return decodingMap;
  102. }
  103. unordered_map<uint8_t, char> Base64::_getEncodingMap()
  104. {
  105. static unordered_map<uint8_t, char> encodingMap = {
  106. {0, 'A'}, {1, 'B'}, {2, 'C'}, {3, 'D'}, {4, 'E'}, {5, 'F'}, {6, 'G'}, {7, 'H'}, {8, 'I'}, {9, 'J'}, {10, 'K'}, {11, 'L'}, {12, 'M'}, {13, 'N'}, {14, 'O'}, {15, 'P'}, {16, 'Q'}, {17, 'R'}, {18, 'S'}, {19, 'T'}, {20, 'U'}, {21, 'V'}, {22, 'W'}, {23, 'X'}, {24, 'Y'}, {25, 'Z'}, {26, 'a'}, {27, 'b'}, {28, 'c'}, {29, 'd'}, {30, 'e'}, {31, 'f'}, {32, 'g'}, {33, 'h'}, {34, 'i'}, {35, 'j'}, {36, 'k'}, {37, 'l'}, {38, 'm'}, {39, 'n'}, {40, 'o'}, {41, 'p'}, {42, 'q'}, {43, 'r'}, {44, 's'}, {45, 't'}, {46, 'u'}, {47, 'v'}, {48, 'w'}, {49, 'x'}, {50, 'y'}, {51, 'z'}, {52, '0'}, {53, '1'}, {54, '2'}, {55, '3'}, {56, '4'}, {57, '5'}, {58, '6'}, {59, '7'}, {60, '8'}, {61, '9'}, {62, '+'}, {63, '/'},
  107. };
  108. return encodingMap;
  109. }
  110. string_view Base64::_getNextByteQuadruple(const string_view _sequence, const size_t _index)
  111. {
  112. return _sequence.substr(_index, 4);
  113. }
  114. string_view Base64::_getNextByteTriple(const string_view _sequence, const size_t _index)
  115. {
  116. return _sequence.substr(_index, 3);
  117. }
  118. void Base64::_mergeBitSequence(uint32_t &_bitStorage, const uint32_t &_bitMask)
  119. {
  120. _bitStorage = _bitStorage | _bitMask;
  121. }
  122. uint32_t Base64::_toDecodingBitStorage(const string_view _quadruple)
  123. {
  124. uint32_t bitStorage{};
  125. uint8_t letterCounter = 1;
  126. unordered_map<char, uint8_t> decodingMap = _getDecodingMap();
  127. for (char letter : _quadruple)
  128. {
  129. uint32_t bitMask = _generateBitMask(decodingMap[letter], (4 - letterCounter) * 6); // must be hardcoded - even in case of less than 4 characters, so that conversion is correct
  130. _mergeBitSequence(bitStorage, bitMask);
  131. ++letterCounter;
  132. }
  133. return bitStorage;
  134. }
  135. uint32_t Base64::_toEncodingBitStorage(const string_view _triple)
  136. {
  137. uint32_t bitStorage{};
  138. uint8_t shiftValue = 16;
  139. for (const char letter : _triple)
  140. {
  141. uint32_t bitMask = _generateBitMask(static_cast<uint8_t>(letter), shiftValue);
  142. _mergeBitSequence(bitStorage, bitMask);
  143. shiftValue -= 8;
  144. }
  145. return bitStorage;
  146. }