|
|
@@ -0,0 +1,165 @@ |
|
|
|
/*
|
|
|
|
==============================================================================
|
|
|
|
|
|
|
|
This file is part of the juce_core module of the JUCE library.
|
|
|
|
Copyright (c) 2013 - Raw Material Software Ltd.
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any purpose with
|
|
|
|
or without fee is hereby granted, provided that the above copyright notice and this
|
|
|
|
permission notice appear in all copies.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
|
|
|
|
TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
|
|
|
|
NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
|
DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
|
|
|
|
IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
|
|
CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
|
|
|
------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
NOTE! This permissive ISC license applies ONLY to files within the juce_core module!
|
|
|
|
All other JUCE modules are covered by a dual GPL/commercial license, so if you are
|
|
|
|
using any other modules, be sure to check that you also comply with their license.
|
|
|
|
|
|
|
|
For more details, visit www.juce.com
|
|
|
|
|
|
|
|
==============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool Base64::convertToBase64 (OutputStream& base64Result, const void* sourceData, size_t sourceDataSize)
|
|
|
|
{
|
|
|
|
static const char lookup[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
|
|
|
const uint8* source = static_cast<const uint8*> (sourceData);
|
|
|
|
|
|
|
|
while (sourceDataSize > 0)
|
|
|
|
{
|
|
|
|
char frame[4];
|
|
|
|
const uint8 byte0 = *source++;
|
|
|
|
frame[0] = lookup [(byte0 & 0xfcu) >> 2];
|
|
|
|
uint32 bits = (byte0 & 0x03u) << 4;
|
|
|
|
|
|
|
|
if (sourceDataSize > 1)
|
|
|
|
{
|
|
|
|
const uint8 byte1 = *source++;
|
|
|
|
frame[1] = lookup[bits | ((byte1 & 0xf0u) >> 4)];
|
|
|
|
bits = (byte1 & 0x0fu) << 2;
|
|
|
|
|
|
|
|
if (sourceDataSize > 2)
|
|
|
|
{
|
|
|
|
const uint8 byte2 = *source++;
|
|
|
|
frame[2] = lookup[bits | ((byte2 & 0xc0u) >> 6)];
|
|
|
|
frame[3] = lookup[byte2 & 0x3fu];
|
|
|
|
sourceDataSize -= 3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
frame[2] = lookup[bits];
|
|
|
|
frame[3] = '=';
|
|
|
|
sourceDataSize = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
frame[1] = lookup[bits];
|
|
|
|
frame[2] = '=';
|
|
|
|
frame[3] = '=';
|
|
|
|
sourceDataSize = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! base64Result.write (frame, 4))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Base64::convertFromBase64 (OutputStream& binaryOutput, StringRef base64TextInput)
|
|
|
|
{
|
|
|
|
for (String::CharPointerType s = base64TextInput.text; ! s.isEmpty();)
|
|
|
|
{
|
|
|
|
uint8 data[4];
|
|
|
|
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
{
|
|
|
|
uint32 c = (uint32) s.getAndAdvance();
|
|
|
|
|
|
|
|
if (c >= 'A' && c <= 'Z') c -= 'A';
|
|
|
|
else if (c >= 'a' && c <= 'z') c -= 'a' - 26;
|
|
|
|
else if (c >= '0' && c <= '9') c += 52 - '0';
|
|
|
|
else if (c == '+') c = 62;
|
|
|
|
else if (c == '/') c = 63;
|
|
|
|
else if (c == '=') { c = 64; if (i <= 1) return false; }
|
|
|
|
else return false;
|
|
|
|
|
|
|
|
data[i] = (uint8) c;
|
|
|
|
}
|
|
|
|
|
|
|
|
binaryOutput.writeByte ((char) ((data[0] << 2) | (data[1] >> 4)));
|
|
|
|
|
|
|
|
if (data[2] < 64)
|
|
|
|
{
|
|
|
|
binaryOutput.writeByte ((char) ((data[1] << 4) | (data[2] >> 2)));
|
|
|
|
|
|
|
|
if (data[3] < 64)
|
|
|
|
binaryOutput.writeByte ((char) ((data[2] << 6) | data[3]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
String Base64::toBase64 (const void* sourceData, size_t sourceDataSize)
|
|
|
|
{
|
|
|
|
MemoryOutputStream m ((sourceDataSize * 4) / 3 + 3);
|
|
|
|
bool ok = convertToBase64 (m, sourceData, sourceDataSize);
|
|
|
|
jassert (ok); // should always succeed for this simple case
|
|
|
|
ignoreUnused (ok);
|
|
|
|
return m.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
String Base64::toBase64 (const String& text)
|
|
|
|
{
|
|
|
|
return toBase64 (text.toRawUTF8(), strlen (text.toRawUTF8()));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
//==============================================================================
|
|
|
|
#if JUCE_UNIT_TESTS
|
|
|
|
|
|
|
|
class Base64Tests : public UnitTest
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Base64Tests() : UnitTest ("Base64 class") {}
|
|
|
|
|
|
|
|
static MemoryBlock createRandomData (Random& r)
|
|
|
|
{
|
|
|
|
MemoryOutputStream m;
|
|
|
|
|
|
|
|
for (int i = r.nextInt (400); --i >= 0;)
|
|
|
|
m.writeByte ((char) r.nextInt (256));
|
|
|
|
|
|
|
|
return m.getMemoryBlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void runTest() override
|
|
|
|
{
|
|
|
|
beginTest ("Base64");
|
|
|
|
|
|
|
|
Random r = getRandom();
|
|
|
|
|
|
|
|
for (int i = 1000; --i >= 0;)
|
|
|
|
{
|
|
|
|
const MemoryBlock original (createRandomData (r));
|
|
|
|
String asBase64 (Base64::toBase64 (original.getData(), original.getSize()));
|
|
|
|
MemoryOutputStream out;
|
|
|
|
expect (Base64::convertFromBase64 (out, asBase64));
|
|
|
|
MemoryBlock result = out.getMemoryBlock();
|
|
|
|
expect (result == original);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static Base64Tests base64Tests;
|
|
|
|
|
|
|
|
#endif
|