Program Listing for File base64.h#

Return to documentation for file (base64.h)

#ifndef BASE64_H
#define BASE64_H

#include <string>
#include <stdexcept>

static const std::string Base64Chars =
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"abcdefghijklmnopqrstuvwxyz"
"0123456789+/";

inline bool IsBase64(const unsigned char c)
{
    return (isalnum(c) || (c == '+') || (c == '/'));
}

inline std::string Base64EncodePlainText(const std::string& plainText)
{
    const char* buffer = plainText.c_str();
    std::size_t bytesLeft = plainText.size();

    std::string ret;
    int i = 0;
    unsigned char char_array_3[3]{ 0 };
    unsigned char char_array_4[4]{ 0 };

    while (bytesLeft--)
    {
        char_array_3[i++] = *reinterpret_cast<const unsigned char*>((buffer++));
        if (i == 3)
        {
            char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;
            char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);
            char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);
            char_array_4[3] = char_array_3[2] & 0x3f;

            for (i = 0; (i < 4); i++)
            {
                ret += Base64Chars[char_array_4[i]];
            }
            i = 0;
        }
    }

    if (i)
    {
        for (int j = i; j < 3; j++)
        {
            char_array_3[j] = '\0';
        }

        char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;
        char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);
        char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);
        char_array_4[3] = char_array_3[2] & 0x3f;

        for (int j = 0; (j < i + 1); j++)
        {
            ret += Base64Chars[char_array_4[j]];
        }

        while ((i++ < 3))
        {
            ret += '=';
        }
    }

    return ret;
}


inline std::string Base64DecodePlainText(const std::string& encoded_string)
{
    size_t in_len = encoded_string.size();
    size_t i = 0;
    size_t in_ = 0;
    unsigned char char_array_4[4], char_array_3[3];
    std::string ret;

    while (in_len-- && (encoded_string[in_] != '=') && IsBase64(encoded_string[in_]))
    {
        char_array_4[i++] = encoded_string[in_]; in_++;
        if (i == 4)
        {
            for (i = 0; i < 4; i++)
            {
                char_array_4[i] = static_cast<uint8_t>(Base64Chars.find(char_array_4[i]));
            }

            char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
            char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
            char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];

            for (i = 0; (i < 3); i++)
            {
                ret+=(reinterpret_cast<char*>(char_array_3)[i]);
            }
            i = 0;
        }
    }

    if (i)
    {
        for (size_t j = i; j < 4; j++)
        {
            char_array_4[j] = 0;
        }

        for (size_t j = 0; j < 4; j++)
        {
            char_array_4[j] = static_cast<uint8_t>(Base64Chars.find(char_array_4[j]));
        }

        char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
        char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
        char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];

        for (size_t j = 0; (j < i - 1); j++)
        {
            ret += (reinterpret_cast<char*>(char_array_3)[j]);
        }
    }
    return ret;
}

inline std::string Base64Encode(const char* data, std::size_t datalength)
{
    const char* buffer = data;
    std::size_t bytesLeft = datalength;

    std::string ret;
    int i = 0;
    unsigned char char_array_3[3]{ 0 };
    unsigned char char_array_4[4]{ 0 };

    while (bytesLeft--)
    {
        char_array_3[i++] = *(buffer++);
        if (i == 3)
        {
            char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;
            char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);
            char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);
            char_array_4[3] = char_array_3[2] & 0x3f;

            for (i = 0; (i < 4); i++)
            {
                ret += Base64Chars[char_array_4[i]];
            }
            i = 0;
        }
    }

    if (i)
    {
        for (int j = i; j < 3; j++)
        {
            char_array_3[j] = '\0';
        }

        char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;
        char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);
        char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);
        char_array_4[3] = char_array_3[2] & 0x3f;

        for (int j = 0; (j < i + 1); j++)
        {
            ret += Base64Chars[char_array_4[j]];
        }

        while ((i++ < 3))
        {
            ret += '=';
        }
    }

    return ret;
}

template <class TConvert>
inline std::string Base64Encode(const TConvert& data)
{
    return Base64Encode(reinterpret_cast<const char*>(&data), sizeof(data));
}

inline std::vector<unsigned char> Base64Decode(const std::string& encoded_string)
{
    size_t in_len = encoded_string.size();
    size_t i = 0;
    size_t in_ = 0;
    unsigned char char_array_4[4], char_array_3[3];
    std::vector<unsigned char> ret;

    while (in_len-- && (encoded_string[in_] != '=') && IsBase64(encoded_string[in_]))
    {
        char_array_4[i++] = encoded_string[in_]; in_++;
        if (i == 4)
        {
            for (i = 0; i < 4; i++)
            {
                char_array_4[i] = static_cast<uint8_t>(Base64Chars.find(char_array_4[i]));
            }

            char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
            char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
            char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];

            for (i = 0; (i < 3); i++)
            {
                ret.push_back(char_array_3[i]);
            }
            i = 0;
        }
    }

    if (i)
    {
        for (size_t j = i; j < 4; j++)
        {
            char_array_4[j] = 0;
        }

        for (size_t j = 0; j < 4; j++)
        {
            char_array_4[j] = static_cast<uint8_t>(Base64Chars.find(char_array_4[j]));
        }

        char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
        char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
        char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];

        for (size_t j = 0; (j < i - 1); j++)
        {
            ret.push_back(char_array_3[j]);
        }
    }
    return ret;
    // memcpy_s(&datablock, sizeof(datablock), ret.data(), ret.size());
}

template <class TConvert>
inline TConvert DecodeBase64(const std::string& encoded_string)
{
    TConvert out{};
    std::vector<unsigned char> rawData = Base64Decode(encoded_string);
    if (sizeof(out) == rawData.size())
    {
        memcpy_s(&out, sizeof(out), rawData.data(), rawData.size());
    }
    else
    {
        TConvert invalid{};
        return invalid;
    }
    return out;
}

template <>
inline std::string DecodeBase64(const std::string& encoded_string)
{
    std::vector<unsigned char> rawData = Base64Decode(encoded_string);
    std::string out(reinterpret_cast<char*>(rawData.data()), rawData.size());
    return out;
}
#endif