2002-04-30 23:03:51 +00:00
|
|
|
/********************************************************************
|
|
|
|
* COPYRIGHT:
|
|
|
|
* Copyright (c) 1997-2001, International Business Machines Corporation and
|
|
|
|
* others. All Rights Reserved.
|
|
|
|
********************************************************************/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* IntlTestCollator is the medium level test class for everything in the directory "collate".
|
|
|
|
*/
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* Modification history
|
|
|
|
* Date Name Description
|
|
|
|
* 02/14/2001 synwee Compare with cintltst and commented away tests
|
|
|
|
* that are not run.
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
#include "unicode/utypes.h"
|
|
|
|
#include "unicode/uchar.h"
|
|
|
|
|
|
|
|
#include "cstring.h"
|
|
|
|
#include "ucol_tok.h"
|
|
|
|
|
|
|
|
#include "tscoll.h"
|
|
|
|
|
|
|
|
#include "dadrcoll.h"
|
|
|
|
|
|
|
|
DataDrivenCollatorTest::DataDrivenCollatorTest()
|
2002-05-10 01:26:46 +00:00
|
|
|
: seq(StringCharacterIterator(""))
|
2002-04-30 23:03:51 +00:00
|
|
|
{
|
|
|
|
UErrorCode status = U_ZERO_ERROR;
|
2002-06-04 08:06:05 +00:00
|
|
|
TestLog testLog;
|
|
|
|
|
|
|
|
driver = TestDataModule::getTestDataModule("DataDrivenCollationTest", testLog, status);
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DataDrivenCollatorTest::~DataDrivenCollatorTest()
|
|
|
|
{
|
2002-05-10 01:26:46 +00:00
|
|
|
delete driver;
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
|
|
|
|
2002-06-17 21:31:41 +00:00
|
|
|
void DataDrivenCollatorTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par */)
|
2002-04-30 23:03:51 +00:00
|
|
|
{
|
2002-06-08 23:31:36 +00:00
|
|
|
if(driver != NULL) {
|
|
|
|
if (exec)
|
|
|
|
{
|
|
|
|
logln("TestSuite Collator: ");
|
2002-06-04 08:06:05 +00:00
|
|
|
}
|
2002-06-08 23:31:36 +00:00
|
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
|
|
const DataMap *info = NULL;
|
|
|
|
TestData *testData = driver->createTestData(index, status);
|
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
name = testData->getName();
|
|
|
|
if(testData->getInfo(info, status)) {
|
|
|
|
log(info->getString("Description", status));
|
|
|
|
}
|
|
|
|
if(exec) {
|
|
|
|
log(name);
|
|
|
|
logln("---");
|
|
|
|
logln("");
|
|
|
|
processTest(testData, status);
|
|
|
|
}
|
|
|
|
delete testData;
|
|
|
|
} else {
|
|
|
|
name = "";
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
2002-06-04 08:06:05 +00:00
|
|
|
} else {
|
2002-06-08 23:31:36 +00:00
|
|
|
errln("collate/DataDrivenTest data not initialized!");
|
2002-06-04 08:06:05 +00:00
|
|
|
name = "";
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
2002-06-04 08:06:05 +00:00
|
|
|
|
|
|
|
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
UBool
|
|
|
|
DataDrivenCollatorTest::setTestSequence(const UnicodeString &setSequence, UnicodeString &source, Collator::EComparisonResult &relation, UErrorCode &status) {
|
|
|
|
seq.setText(setSequence);
|
|
|
|
return getNextInSequence(source, relation, status);
|
|
|
|
}
|
|
|
|
|
2002-05-01 18:33:04 +00:00
|
|
|
// Parses the sequence to be tested
|
2002-04-30 23:03:51 +00:00
|
|
|
UBool
|
|
|
|
DataDrivenCollatorTest::getNextInSequence(UnicodeString &source, Collator::EComparisonResult &relation, UErrorCode &status) {
|
|
|
|
source.truncate(0);
|
|
|
|
// TODO: add quoting support - will need it pretty soon!
|
|
|
|
UBool quoted = FALSE;
|
|
|
|
UBool quotedsingle = FALSE;
|
|
|
|
UChar32 currChar = 0;
|
|
|
|
|
|
|
|
while(currChar != CharacterIterator::DONE) {
|
|
|
|
currChar= seq.next32PostInc();
|
|
|
|
if(!quoted) {
|
|
|
|
if(u_isWhitespace(currChar)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
switch(currChar) {
|
|
|
|
case CharacterIterator::DONE:
|
|
|
|
break;
|
|
|
|
case 0x003C /* < */:
|
|
|
|
relation = Collator::LESS;
|
|
|
|
currChar = CharacterIterator::DONE;
|
|
|
|
break;
|
|
|
|
case 0x003D /* = */:
|
|
|
|
relation = Collator::EQUAL;
|
|
|
|
currChar = CharacterIterator::DONE;
|
|
|
|
break;
|
|
|
|
case 0x003E /* > */:
|
|
|
|
relation = Collator::GREATER;
|
|
|
|
currChar = CharacterIterator::DONE;
|
|
|
|
break;
|
|
|
|
case 0x0027 /* ' */: /* very basic quoting */
|
|
|
|
quoted = TRUE;
|
|
|
|
quotedsingle = FALSE;
|
|
|
|
break;
|
|
|
|
case 0x005c /* \ */: /* single quote */
|
|
|
|
quoted = TRUE;
|
|
|
|
quotedsingle = TRUE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
source.append(currChar);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if(currChar == CharacterIterator::DONE) {
|
|
|
|
status = U_ILLEGAL_ARGUMENT_ERROR;
|
|
|
|
errln("Quote in sequence not closed!");
|
|
|
|
return FALSE;
|
|
|
|
} else if(currChar == 0x0027) {
|
|
|
|
quoted = FALSE;
|
|
|
|
} else {
|
|
|
|
source.append(currChar);
|
|
|
|
}
|
|
|
|
if(quotedsingle) {
|
|
|
|
quoted = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return seq.hasNext();
|
|
|
|
}
|
|
|
|
|
2002-05-01 18:33:04 +00:00
|
|
|
// Reads the options string and sets appropriate attributes in collator
|
2002-04-30 23:03:51 +00:00
|
|
|
void
|
2002-05-01 18:33:04 +00:00
|
|
|
DataDrivenCollatorTest::processArguments(Collator *col, const UChar *start, int32_t optLen, UErrorCode &status) {
|
|
|
|
const UChar *end = start+optLen;
|
2002-04-30 23:03:51 +00:00
|
|
|
UColAttribute attrib;
|
|
|
|
UColAttributeValue value;
|
|
|
|
|
2002-05-10 01:26:46 +00:00
|
|
|
if(optLen == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2002-04-30 23:03:51 +00:00
|
|
|
start = ucol_tok_getNextArgument(start, end, &attrib, &value, &status);
|
|
|
|
while(start != NULL) {
|
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
col->setAttribute(attrib, value, status);
|
|
|
|
}
|
|
|
|
start = ucol_tok_getNextArgument(start, end, &attrib, &value, &status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-06-04 08:06:05 +00:00
|
|
|
DataDrivenCollatorTest::processTest(TestData *testData, UErrorCode &status) {
|
2002-04-30 23:03:51 +00:00
|
|
|
Collator *col = NULL;
|
2002-06-04 08:06:05 +00:00
|
|
|
/*
|
2002-05-10 01:26:46 +00:00
|
|
|
UnicodeString testInit[256];
|
|
|
|
const char **testNames = new const char*[256];
|
|
|
|
int32_t settingsSetSize = 0;
|
|
|
|
const UChar *arguments = NULL;
|
|
|
|
int32_t argLen = 0;
|
|
|
|
int32_t i = 0;
|
|
|
|
while(settingsSetSize = driver->getNextSettingsSet(testNames, testInit, 256, status)) {
|
|
|
|
argLen = 0;
|
|
|
|
for(i = 0; i < settingsSetSize; i++) {
|
|
|
|
if(strcmp(testNames[i], "Locale") == 0) { // Make the locale dependent collator
|
|
|
|
if(col == NULL) {
|
|
|
|
char localeName[256];
|
|
|
|
testInit[i].extract(0, testInit[i].length(), localeName, "");
|
|
|
|
col = Collator::createInstance(localeName, status);
|
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
logln("Testing collator for locale "+testInit[i]);
|
|
|
|
} else {
|
|
|
|
errln("Unable to instantiate collator for locale "+testInit[i]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
errln("Collator defined more than once!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if(strcmp(testNames[i], "Rules") == 0) {
|
|
|
|
if(col == NULL) {
|
|
|
|
col = new RuleBasedCollator(testInit[i], status);
|
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
logln("Testing collator for rules "+UnicodeString(testInit[i]));
|
|
|
|
} else {
|
|
|
|
errln("Unable to instantiate collator for rules "+UnicodeString(testInit[i]));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
errln("Collator defined more than once!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if(strcmp(testNames[i], "Attributes") == 0) {
|
|
|
|
logln("Arguments: "+testInit[i]);
|
|
|
|
argLen = testInit[i].length();
|
|
|
|
arguments = testInit[i].getBuffer();
|
2002-04-30 23:03:51 +00:00
|
|
|
} else {
|
2002-05-10 01:26:46 +00:00
|
|
|
errln("I don't understand the setting "+UnicodeString(testNames[i]));
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
|
|
|
}
|
2002-05-10 01:26:46 +00:00
|
|
|
if(col != NULL) {
|
|
|
|
if(argLen > 0) {
|
|
|
|
processArguments(col, arguments, argLen, status);
|
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
UnicodeString sequence[1];
|
|
|
|
while(driver->getNextTestCase(sequence, 1, status)) {
|
|
|
|
processSequence(col, *sequence, status);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
errln("Couldn't process arguments");
|
|
|
|
}
|
2002-05-01 18:33:04 +00:00
|
|
|
}
|
2002-05-10 01:26:46 +00:00
|
|
|
} else {
|
|
|
|
errln("Couldn't instantiate a collator!");
|
2002-05-01 18:33:04 +00:00
|
|
|
}
|
2002-05-10 01:26:46 +00:00
|
|
|
delete col;
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
2002-05-10 01:26:46 +00:00
|
|
|
delete testNames;
|
2002-06-04 08:06:05 +00:00
|
|
|
*/
|
|
|
|
const UChar *arguments = NULL;
|
|
|
|
int32_t argLen = 0;
|
|
|
|
const DataMap *settings = NULL;
|
|
|
|
const DataMap *currentCase = NULL;
|
|
|
|
UErrorCode intStatus = U_ZERO_ERROR;
|
|
|
|
UnicodeString testSetting;
|
|
|
|
while(testData->nextSettings(settings, status)) {
|
2002-07-02 22:37:47 +00:00
|
|
|
intStatus = U_ZERO_ERROR;
|
2002-06-04 08:06:05 +00:00
|
|
|
// try to get a locale
|
|
|
|
testSetting = settings->getString("TestLocale", intStatus);
|
|
|
|
if(U_SUCCESS(intStatus)) {
|
|
|
|
char localeName[256];
|
|
|
|
testSetting.extract(0, testSetting.length(), localeName, "");
|
|
|
|
col = Collator::createInstance(localeName, status);
|
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
logln("Testing collator for locale "+testSetting);
|
|
|
|
} else {
|
|
|
|
errln("Unable to instantiate collator for locale "+testSetting);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// if no locale, try from rules
|
|
|
|
intStatus = U_ZERO_ERROR;
|
|
|
|
testSetting = settings->getString("Rules", intStatus);
|
|
|
|
if(U_SUCCESS(intStatus)) {
|
|
|
|
col = new RuleBasedCollator(testSetting, status);
|
2002-07-02 22:37:47 +00:00
|
|
|
UCollator *ucol = ucol_openRules(testSetting.getBuffer(), testSetting.length(), UCOL_DEFAULT, UCOL_DEFAULT, NULL, &status);
|
2002-06-04 08:06:05 +00:00
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
logln("Testing collator for rules "+testSetting);
|
|
|
|
} else {
|
|
|
|
errln("Unable to instantiate collator for rules "+testSetting);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
errln("No collator definition!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(col != NULL) {
|
|
|
|
// get attributes
|
|
|
|
testSetting = settings->getString("Arguments", intStatus);
|
|
|
|
if(U_SUCCESS(intStatus)) {
|
|
|
|
logln("Arguments: "+testSetting);
|
|
|
|
argLen = testSetting.length();
|
|
|
|
arguments = testSetting.getBuffer();
|
|
|
|
processArguments(col, arguments, argLen, intStatus);
|
|
|
|
if(U_FAILURE(intStatus)) {
|
|
|
|
errln("Couldn't process arguments");
|
|
|
|
break;
|
|
|
|
}
|
2002-07-02 22:37:47 +00:00
|
|
|
} else {
|
|
|
|
intStatus = U_ZERO_ERROR;
|
2002-06-04 08:06:05 +00:00
|
|
|
}
|
|
|
|
// Start the processing
|
|
|
|
while(testData->nextCase(currentCase, status)) {
|
|
|
|
UnicodeString sequence = currentCase->getString("sequence", status);
|
|
|
|
if(U_SUCCESS(status)) {
|
|
|
|
processSequence(col, sequence, status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
errln("Couldn't instantiate a collator!");
|
|
|
|
}
|
|
|
|
delete col;
|
2002-07-02 22:37:47 +00:00
|
|
|
col = NULL;
|
2002-06-04 08:06:05 +00:00
|
|
|
}
|
2002-04-30 23:03:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
DataDrivenCollatorTest::processSequence(Collator* col, const UnicodeString &sequence, UErrorCode &status) {
|
|
|
|
UnicodeString source;
|
|
|
|
UnicodeString target;
|
|
|
|
UnicodeString temp;
|
2002-06-13 20:43:03 +00:00
|
|
|
Collator::EComparisonResult relation = Collator::EQUAL;
|
|
|
|
Collator::EComparisonResult nextRelation = Collator::EQUAL;
|
2002-04-30 23:03:51 +00:00
|
|
|
UBool hasNext;
|
|
|
|
|
|
|
|
setTestSequence(sequence, source, relation, status);
|
|
|
|
|
|
|
|
// TODO: have a smarter tester that remembers the sequence and ensures that
|
|
|
|
// the complete sequence is in order. That is why I have made a constraint
|
|
|
|
// in the sequence format.
|
|
|
|
do {
|
|
|
|
hasNext = getNextInSequence(target, nextRelation, status);
|
|
|
|
doTest(col, source, target, relation);
|
|
|
|
source = target;
|
|
|
|
relation = nextRelation;
|
|
|
|
} while(hasNext);
|
|
|
|
}
|
|
|
|
|