Added correct base64 encode/decode implementation to use in API

This commit is contained in:
Alex Zolotarev 2012-12-15 16:17:56 +03:00 committed by Alex Zolotarev
parent 98372ba583
commit 5be7d460ae
5 changed files with 76 additions and 37 deletions

View file

@ -11,6 +11,9 @@ DEFINES += LTC_NO_ROLC
SOURCES += \
src/hashes/sha2/sha256.c \
src/misc/base64/base64_decode.c \
src/misc/base64/base64_encode.c \
HEADERS += \
src/headers/tomcrypt.h \
src/headers/tomcrypt_misc.h \

View file

@ -1,5 +1,9 @@
#include "base64.hpp"
#include "../3party/tomcrypt/src/headers/tomcrypt.h"
#include "../3party/tomcrypt/src/headers/tomcrypt_misc.h"
#include "../base/assert.hpp"
#include <boost/archive/iterators/base64_from_binary.hpp>
#include <boost/archive/iterators/binary_from_base64.hpp>
#include <boost/archive/iterators/transform_width.hpp>
@ -35,3 +39,44 @@ namespace base64_for_user_ids
return string(binary_t(base64Chars.begin()), binary_t(base64Chars.begin() + base64Chars.size() - 1));
}
}
namespace base64
{
string Encode(string const & bytesToEncode)
{
string result;
long unsigned int resSize = (bytesToEncode.size() + 3) * 4 / 3;
// Raw new/delete because we don't throw exceptions inbetween
unsigned char * buffer = new unsigned char[resSize];
if (CRYPT_OK == base64_encode(reinterpret_cast<unsigned char const *>(bytesToEncode.data()),
bytesToEncode.size(),
buffer,
&resSize))
result.assign(reinterpret_cast<char const *>(buffer), resSize);
else
ASSERT(false, ("It should work!"));
delete[] buffer;
return result;
}
string Decode(string const & base64CharsToDecode)
{
string result;
long unsigned int resSize = base64CharsToDecode.size() * 3 / 4 + 2;
// Raw new/delete because we don't throw exceptions inbetween
unsigned char * buffer = new unsigned char[resSize];
if (CRYPT_OK == base64_decode(reinterpret_cast<unsigned char const *>(base64CharsToDecode.data()),
base64CharsToDecode.size(),
buffer,
&resSize))
result.assign(reinterpret_cast<char const *>(buffer), resSize);
else
ASSERT(false, ("It should work!"));
delete[] buffer;
return result;
}
}

View file

@ -1,7 +1,16 @@
#pragma once
#include "../std/vector.hpp"
#include "../std/string.hpp"
namespace base64
{
string Encode(string const & bytesToEncode);
string Decode(string const & base64CharsToDecode);
}
/// This namespace contains historically invalid implementation of base64,
/// but it's still needed for production code
namespace base64_for_user_ids

View file

@ -5,46 +5,27 @@
#include "../base64.hpp"
UNIT_TEST(Base64_Encode)
{
TEST_EQUAL(base64::encode("Hello, world!"), "SGVsbG8sIHdvcmxkITAw", ());
TEST_EQUAL(base64::encode(""), "", ());
TEST_EQUAL(base64::encode("$"), "JDAw", ());
TEST_EQUAL(base64::encode("MapsWithMe is an offline maps application for any device in the world."),
"TWFwc1dpdGhNZSBpcyBhbiBvZmZsaW5lIG1hcHMgYXBwbGljYXRpb24gZm9yIGFueSBkZXZpY2Ug"
"aW4gdGhlIHdvcmxkLjAw", ());
}
UNIT_TEST(Base64_Decode)
{
TEST_EQUAL(base64::decode("SGVsbG8sIHdvcmxkIQ"), "Hello, world!", ());
TEST_EQUAL(base64::decode(""), "", ());
TEST_EQUAL(base64::decode("JA"), "$", ());
TEST_EQUAL(base64::decode("TWFwc1dpdGhNZSBpcyBhbiBvZmZsaW5lIG1hcHMgYXBwbGljYXRpb24gZm9yIGFueSBkZXZpY2Ug"
"aW4gdGhlIHdvcmxkLg"),
"MapsWithMe is an offline maps application for any device in the world.", ());
}
using namespace base64;
UNIT_TEST(Base64_QualityTest)
UNIT_TEST(Base64_Smoke)
{
size_t const NUMBER_OF_TESTS = 10000;
LCG32 generator(NUMBER_OF_TESTS);
for (size_t i = 0; i < NUMBER_OF_TESTS; ++i)
char const * bytes[] = {"H", "He", "Hel", "Hell", "Hello", "Hello,", "Hello, ", "Hello, World!"};
char const * encoded[] = {"SA==", "SGU=", "SGVs", "SGVsbA==", "SGVsbG8=", "SGVsbG8s",
"SGVsbG8sIA==", "SGVsbG8sIFdvcmxkIQ=="};
TEST_EQUAL(ARRAY_SIZE(bytes), ARRAY_SIZE(encoded), ());
for (size_t i = 0; i < ARRAY_SIZE(bytes); ++i)
{
string randomBytes;
for (size_t j = 0 ; j < 8; ++j)
{
if (j == 4)
{
randomBytes.push_back('\0');
continue;
}
randomBytes.push_back(static_cast<char>(generator.Generate()));
}
string const result = base64::encode(randomBytes);
TEST_GREATER_OR_EQUAL(result.size(), randomBytes.size(),
(randomBytes, result));
for (size_t i = 0; i < result.size(); ++i)
TEST_NOT_EQUAL(result[i], 0, ());
TEST_EQUAL(Encode(bytes[i]), encoded[i], ());
TEST_EQUAL(Decode(encoded[i]), bytes[i], ());
TEST_EQUAL(Decode(Encode(bytes[i])), bytes[i], ());
TEST_EQUAL(Encode(Decode(encoded[i])), encoded[i], ());
}
char const * str = "MapsWithMe is the offline maps application for any device in the world.";
char const * encStr = "TWFwc1dpdGhNZSBpcyB0aGUgb2ZmbGluZSBtYXBzIGFwcGxpY2F0aW9uIGZvciBhbnkgZGV2aWNlIGluIHRoZSB3b3JsZC4=";
TEST_EQUAL(Encode(str), encStr, ());
TEST_EQUAL(Decode(encStr), str, ());
}

View file

@ -32,6 +32,7 @@ SOURCES += ../../testing/testingmain.cpp \
gzip_test.cpp \
coder_util_test.cpp \
bit_shift_test.cpp \
base64_test.cpp \
base64_for_user_id_test.cpp \
sha2_test.cpp \
value_opt_string_test.cpp \