mirror of
https://github.com/qpdf/qpdf.git
synced 2025-01-23 07:08:30 +00:00
154 lines
3.1 KiB
C++
154 lines
3.1 KiB
C++
#include <qpdf/MD5.hh>
|
|
|
|
#include <qpdf/QIntC.hh>
|
|
#include <qpdf/QPDFCryptoProvider.hh>
|
|
#include <qpdf/QUtil.hh>
|
|
|
|
#include <cstring>
|
|
|
|
MD5::MD5()
|
|
{
|
|
init();
|
|
}
|
|
|
|
void
|
|
MD5::init()
|
|
{
|
|
this->crypto = QPDFCryptoProvider::getImpl();
|
|
this->crypto->MD5_init();
|
|
}
|
|
|
|
void
|
|
MD5::finalize()
|
|
{
|
|
this->crypto->MD5_finalize();
|
|
}
|
|
|
|
void
|
|
MD5::reset()
|
|
{
|
|
init();
|
|
}
|
|
|
|
void
|
|
MD5::encodeString(char const* str)
|
|
{
|
|
size_t len = strlen(str);
|
|
crypto->MD5_init();
|
|
encodeDataIncrementally(str, len);
|
|
crypto->MD5_finalize();
|
|
}
|
|
|
|
void
|
|
MD5::appendString(char const* input_string)
|
|
{
|
|
encodeDataIncrementally(input_string, strlen(input_string));
|
|
}
|
|
|
|
void
|
|
MD5::encodeDataIncrementally(char const* data, size_t len)
|
|
{
|
|
this->crypto->MD5_update(QUtil::unsigned_char_pointer(data), len);
|
|
}
|
|
|
|
void
|
|
MD5::encodeFile(char const* filename, qpdf_offset_t up_to_offset)
|
|
{
|
|
char buffer[1024];
|
|
|
|
FILE* file = QUtil::safe_fopen(filename, "rb");
|
|
size_t len;
|
|
size_t so_far = 0;
|
|
size_t to_try = 1024;
|
|
size_t up_to_size = 0;
|
|
if (up_to_offset >= 0) {
|
|
up_to_size = QIntC::to_size(up_to_offset);
|
|
}
|
|
do {
|
|
if ((up_to_offset >= 0) && ((so_far + to_try) > up_to_size)) {
|
|
to_try = up_to_size - so_far;
|
|
}
|
|
len = fread(buffer, 1, to_try, file);
|
|
if (len > 0) {
|
|
encodeDataIncrementally(buffer, len);
|
|
so_far += len;
|
|
if ((up_to_offset >= 0) && (so_far >= up_to_size)) {
|
|
break;
|
|
}
|
|
}
|
|
} while (len > 0);
|
|
if (ferror(file)) {
|
|
// Assume, perhaps incorrectly, that errno was set by the underlying call to read....
|
|
(void)fclose(file);
|
|
QUtil::throw_system_error(std::string("MD5: read error on ") + filename);
|
|
}
|
|
(void)fclose(file);
|
|
|
|
this->crypto->MD5_finalize();
|
|
}
|
|
|
|
void
|
|
MD5::digest(Digest result)
|
|
{
|
|
this->crypto->MD5_finalize();
|
|
this->crypto->MD5_digest(result);
|
|
}
|
|
|
|
void
|
|
MD5::print()
|
|
{
|
|
Digest digest_val;
|
|
digest(digest_val);
|
|
|
|
unsigned int i;
|
|
for (i = 0; i < 16; ++i) {
|
|
printf("%02x", digest_val[i]);
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
std::string
|
|
MD5::unparse()
|
|
{
|
|
this->crypto->MD5_finalize();
|
|
Digest digest_val;
|
|
digest(digest_val);
|
|
return QUtil::hex_encode(std::string(reinterpret_cast<char*>(digest_val), 16));
|
|
}
|
|
|
|
std::string
|
|
MD5::getDataChecksum(char const* buf, size_t len)
|
|
{
|
|
MD5 m;
|
|
m.encodeDataIncrementally(buf, len);
|
|
return m.unparse();
|
|
}
|
|
|
|
std::string
|
|
MD5::getFileChecksum(char const* filename, qpdf_offset_t up_to_offset)
|
|
{
|
|
MD5 m;
|
|
m.encodeFile(filename, up_to_offset);
|
|
return m.unparse();
|
|
}
|
|
|
|
bool
|
|
MD5::checkDataChecksum(char const* const checksum, char const* buf, size_t len)
|
|
{
|
|
std::string actual_checksum = getDataChecksum(buf, len);
|
|
return (checksum == actual_checksum);
|
|
}
|
|
|
|
bool
|
|
MD5::checkFileChecksum(char const* const checksum, char const* filename, qpdf_offset_t up_to_offset)
|
|
{
|
|
bool result = false;
|
|
try {
|
|
std::string actual_checksum = getFileChecksum(filename, up_to_offset);
|
|
result = (checksum == actual_checksum);
|
|
} catch (std::runtime_error const&) {
|
|
// Ignore -- return false
|
|
}
|
|
return result;
|
|
}
|