Base64.cpp 5.1 KB

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