/* This file is part of the KDE libraries Copyright (C) 2000,2001 Dawit Alemayehu <adawit@kde.org> This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include <config.h> #include <unistd.h> #include <time.h> #include <iostream> #include <tqbuffer.h> #include <tqfile.h> #include <kdebug.h> #include <klocale.h> #include <kcmdlineargs.h> #include <kapplication.h> #include <kmdcodec.h> using namespace std; #define TEST_BLOCK_LEN 1000 // Length of test blocks. #define TEST_BLOCK_COUNT 10000 // Number of test blocks. #define MAX_READ_BUF_SIZE 8192 enum Codec { Unspecified=0, Base64Encode, Base64Decode, UUEncode, UUDecode, QPEncode, QPDecode }; void MD5_timeTrial (); void MD5_testSuite (); void testCodec (const char*, Codec, bool); void MD5_verify (const char*, const char*, bool); void MD5_file (const char * , bool rawOutput = false); void MD5_string (const char *, const char *expected = 0, bool rawOutput = false); long readContent (const TQFile& f, long count, TQByteArray& buf) { long result; int old_size; old_size = buf.size(); buf.resize(old_size+count); result = read (f.handle (), buf.data()+old_size, count); if ( result > 0 && result < count ) { buf.resize( old_size + result ); } else if ( result == 0 ) { buf.resize( old_size ); } else if ( result == -1 ) { kdError() << "Could not read the file!" << endl; } return result; } void testCodec (const char* msg, Codec type, bool isFile) { TQByteArray output; if ( isFile ) { int count; TQByteArray data; TQFile f (TQFile::encodeName(msg)); if (!f.exists()) { kdError() << "Could not find: " << f.name () << endl; return; } if (!f.open(IO_ReadOnly)) { f.close (); kdError() << "Could not open: " << f.name() << endl; return; } // Read contents of file... count = 0; while ((count= readContent(f, MAX_READ_BUF_SIZE, data)) > 0); // Error! Exit! if ( count == -1 ) { kdError () << "Error reading from: " << f.name() << endl; f.close (); return; } f.close (); // Perform the requested encoding or decoding... switch (type) { case Base64Encode: KCodecs::base64Encode(data, output, true); break; case Base64Decode: KCodecs::base64Decode(data, output); break; case UUEncode: KCodecs::uuencode(data, output); break; case UUDecode: KCodecs::uudecode(data, output); break; case QPEncode: KCodecs::quotedPrintableEncode(data, output, true); break; case QPDecode: KCodecs::quotedPrintableDecode(data, output); break; default: break; } TQCString result (output.data(), output.size()+1); cout << "Result: " << endl << result << endl; } else { TQCString result; const size_t len = strlen(msg); output.resize(len); memcpy (output.data(), msg, len); switch (type) { case Base64Encode: result = KCodecs::base64Encode(output); break; case Base64Decode: result = KCodecs::base64Decode(output); break; case UUEncode: result = KCodecs::uuencode(output); break; case UUDecode: result = KCodecs::uudecode(output); break; case QPEncode: result = KCodecs::quotedPrintableEncode(output); break; case QPDecode: result = KCodecs::quotedPrintableDecode(output); break; default: break; } cout << result << endl; } } void MD5_timeTrial () { KMD5 context; time_t endTime; time_t startTime; TQ_UINT8 block[TEST_BLOCK_LEN]; TQ_UINT32 i; cout << "Timing test. Digesting " << TEST_BLOCK_COUNT << " blocks of " << TEST_BLOCK_LEN << "-byte..." << endl; // Initialize block for (i = 0; i < TEST_BLOCK_LEN; i++) block[i] = (TQ_UINT8)(i & 0xff); // Start timer time (&startTime); // Digest blocks for (i = 0; i < TEST_BLOCK_COUNT; i++) context.update (block, TEST_BLOCK_LEN); // Stop timer time (&endTime); long duration = endTime - startTime; long speed; if (duration) speed = (TEST_BLOCK_LEN * (TEST_BLOCK_COUNT/duration)); else speed = TEST_BLOCK_COUNT; cout << "Result: " << endl; cout << " Time = " << duration << " seconds" << endl; cout << " Speed = " << speed << " bytes/second" << endl; cout << " Digest = " << context.hexDigest() << endl; } void MD5_testSuite () { cout << "MD5 preset test suite as defined in RFC 1321:" << endl; MD5_string ( "", "d41d8cd98f00b204e9800998ecf8427e" ); MD5_string ( "a", "0cc175b9c0f1b6a831c399e269772661" ); MD5_string ( "abc", "900150983cd24fb0d6963f7d28e17f72" ); MD5_string ( "message digest", "f96b697d7cb7938d525a2f31aaf161d0" ); MD5_string ( "abcdefghijklmnopqrstuvwxyz", "c3fcd3d76192e4007dfb496cca67e13b" ); MD5_string ( "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789", "d174ab98d277d9f5a5611c2c9f419d9f" ); MD5_string ( "12345678901234567890123456789012345678901234567890123456789012" "345678901234567890", "57edf4a22be3c955ac49da2e2107b67a" ); } void MD5_verify( const char *input, const char *digest, bool isFile ) { bool result; KMD5 context; if ( !isFile ) { context.update (TQCString(input)); result = context.verify( digest ); cout << "Input string: " << input << endl; } else { TQFile f (input); if (!f.open (IO_ReadOnly)) { f.close (); kdFatal() << "Cannot open file for reading!" << endl; } result = context.verify (digest); f.close (); cout << "Input filename: " << input << endl; } cout << "Calculated Digest = " << context.hexDigest() << endl; cout << "Supplied Digest = " << digest << endl; cout << "Matches: " << (result ? "TRUE":"FALSE") << endl; } void MD5_file (const char *filename, bool rawOutput ) { TQFile f (TQFile::encodeName(filename)); if (!f.open(IO_ReadOnly)) { f.close(); kdError() << "(" << filename << ") cannot be opened!" << endl; return; } KMD5 context; context.update( f ); if ( rawOutput ) cout << "MD5 (" << filename << ") = " << context.rawDigest() << endl; else cout << "MD5 (" << filename << ") = " << context.hexDigest() << endl; f.close (); } void MD5_string (const char *input, const char* expected, bool rawOutput ) { KMD5 context; context.update (TQCString(input)); cout << "Checking MD5 for: " << input << endl; if ( rawOutput ) cout << "Result: " << context.rawDigest() << endl; else cout << "Result: " << context.hexDigest() << endl; if ( expected ) { cout << "Expected: " << expected << endl; cout << "Status: " << context.verify (expected) << endl; } } int main (int argc, char *argv[]) { const char *version = "1.0"; const char *description = "Unit test for md5, base64 encode/decode and uuencode/decode facilities"; KCmdLineOptions options[] = { { "c <digest>", "compare <digest> with the calculated digest for a string or file.", 0 }, { "d", "decode the given string or file using base64", 0 }, { "e", "encode the given string or file using base64", 0 }, { "f", "the filename to be used as input", "default" }, { "p", "encode the given string or file using quoted-printable", 0}, { "q", "decode the given string or file using quoted-printable", 0}, { "r", "calculate the raw md5 for the given string or file", 0 }, { "s", "the string to be used as input", 0 }, { "t", "perform a timed message-digest test", 0 }, { "u", "uuencode the given string or file", 0 }, { "x", "uudecode the given string or file", 0 }, { "z", "run a preset message-digest test", 0 }, { "+command", "[input1, input2,...]", 0 }, KCmdLineLastOption }; TDECmdLineArgs::init( argc, argv, "kmdcodectest", description, version ); TDECmdLineArgs::addCmdLineOptions( options ); TDECmdLineArgs *args = TDECmdLineArgs::parsedArgs(); int count = args->count(); TDEApplication app; if (!count) { if ( args->isSet("t") ) MD5_timeTrial (); else if ( args->isSet("z") ) MD5_testSuite (); else args->usage(); } else { bool isVerify = args->isSet("c"); bool isString = args->isSet("s"); bool isFile = args->isSet( "f" ); Codec type = Unspecified; if ( args->isSet("d") ) type = Base64Decode; else if ( args->isSet("e") ) type = Base64Encode; else if ( args->isSet("u") ) type = UUEncode; else if ( args->isSet("x") ) type = UUDecode; else if ( args->isSet("p") ) type = QPEncode; else if ( args->isSet("q") ) type = QPDecode; if ( isVerify ) { const char* opt = args->getOption( "c" ).data(); for ( int i=0 ; i < count; i++ ) MD5_verify ( TQCString(args->arg(i)), opt, (isString || !isFile) ); } else { for ( int i=0 ; i < count; i++ ) { if ( type != Unspecified ) testCodec( args->arg(i), type, isFile ); else { if ( isString ) MD5_string( args->arg( i ), 0, args->isSet("r") ); else MD5_file( args->arg( i ), args->isSet("r") ); } } } } args->clear(); return (0); }