ce7e060d50
This makes fixes in order to run the icu4c tests (intltest, cintltst, iotest, and icuinfo) cleanly under valgrind with --leak-check=full.
485 lines
17 KiB
C++
485 lines
17 KiB
C++
// © 2016 and later: Unicode, Inc. and others.
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
|
/********************************************************************
|
|
* COPYRIGHT:
|
|
* Copyright (c) 1997-2010, International Business Machines Corporation and
|
|
* others. All Rights Reserved.
|
|
********************************************************************/
|
|
|
|
/***********************************************************************
|
|
* Modification history
|
|
* Date Name Description
|
|
* 07/09/2007 srl Copied from dadrcoll.cpp
|
|
***********************************************************************/
|
|
|
|
#include "unicode/utypes.h"
|
|
|
|
#if !UCONFIG_NO_FORMATTING
|
|
|
|
#include "unicode/tstdtmod.h"
|
|
#include "tsdate.h"
|
|
#include "dadrcal.h"
|
|
#include "unicode/calendar.h"
|
|
#include "intltest.h"
|
|
#include <string.h>
|
|
#include "unicode/schriter.h"
|
|
#include "unicode/regex.h"
|
|
#include "unicode/smpdtfmt.h"
|
|
#include "dbgutil.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
DataDrivenCalendarTest::DataDrivenCalendarTest() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
driver = TestDataModule::getTestDataModule("calendar", *this, status);
|
|
}
|
|
|
|
DataDrivenCalendarTest::~DataDrivenCalendarTest() {
|
|
delete driver;
|
|
}
|
|
|
|
void DataDrivenCalendarTest::runIndexedTest(int32_t index, UBool exec,
|
|
const char* &name, char* /*par */) {
|
|
if (driver != NULL) {
|
|
if (exec) {
|
|
// logln("Begin ");
|
|
}
|
|
const DataMap *info= NULL;
|
|
UErrorCode status= U_ZERO_ERROR;
|
|
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);
|
|
}
|
|
delete testData;
|
|
} else {
|
|
name = "";
|
|
}
|
|
} else {
|
|
dataerrln("format/DataDriven*Test data (calendar.res) not initialized!");
|
|
name = "";
|
|
}
|
|
|
|
}
|
|
|
|
void DataDrivenCalendarTest::testOps(TestData *testData,
|
|
const DataMap * /*settings*/) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UBool useDate = FALSE; // TODO
|
|
UnicodeString kMILLIS("MILLIS="); // TODO: static
|
|
UDate fromDate = 0; // TODO
|
|
UDate toDate = 0;
|
|
|
|
const DataMap *currentCase= NULL;
|
|
char toCalLoc[256] = "";
|
|
|
|
// TODO: static strings?
|
|
const UnicodeString kADD("add", "");
|
|
const UnicodeString kROLL("roll", "");
|
|
|
|
// Get 'from' time
|
|
CalendarFieldsSet fromSet, toSet, paramsSet, diffSet;
|
|
SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy / YYYY'-W'ww-ee"),
|
|
status);
|
|
if (U_FAILURE(status)) {
|
|
dataerrln("FAIL: Couldn't create SimpleDateFormat: %s",
|
|
u_errorName(status));
|
|
return;
|
|
}
|
|
// Start the processing
|
|
int n = 0;
|
|
while (testData->nextCase(currentCase, status)) {
|
|
++n;
|
|
Calendar *toCalendar= NULL;
|
|
Calendar *fromCalendar= NULL;
|
|
|
|
// load parameters
|
|
char theCase[200];
|
|
sprintf(theCase, "[case %d]", n);
|
|
UnicodeString caseString(theCase, "");
|
|
// build to calendar
|
|
// Headers { "locale","from","operation","params","to" }
|
|
// #1 locale
|
|
const char *param = "locale";
|
|
UnicodeString locale;
|
|
UnicodeString testSetting = currentCase->getString(param, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Unable to get param '"+param+"' "
|
|
+ UnicodeString(" - "));
|
|
continue;
|
|
}
|
|
testSetting.extract(0, testSetting.length(), toCalLoc, (const char*)0);
|
|
fromCalendar = Calendar::createInstance(toCalLoc, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Unable to instantiate calendar for "
|
|
+testSetting);
|
|
continue;
|
|
}
|
|
|
|
fromSet.clear();
|
|
// #2 'from' info
|
|
param = "from";
|
|
UnicodeString from = testSetting=currentCase->getString(param, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Unable to get parameter '"+param+"' "
|
|
+ UnicodeString(" - "));
|
|
continue;
|
|
}
|
|
|
|
if(from.startsWith(kMILLIS)){
|
|
UnicodeString millis = UnicodeString(from, kMILLIS.length());
|
|
useDate = TRUE;
|
|
fromDate = udbg_stod(millis);
|
|
} else if(fromSet.parseFrom(testSetting, status)<0 || U_FAILURE(status)){
|
|
errln(caseString+": Failed to parse '"+param+"' parameter: "
|
|
+testSetting);
|
|
continue;
|
|
}
|
|
|
|
// #4 'operation' info
|
|
param = "operation";
|
|
UnicodeString operation = testSetting=currentCase->getString(param,
|
|
status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Unable to get parameter '"+param+"' "
|
|
+ UnicodeString(" - "));
|
|
continue;
|
|
}
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Failed to parse '"+param+"' parameter: "
|
|
+testSetting);
|
|
continue;
|
|
}
|
|
|
|
paramsSet.clear();
|
|
// #3 'params' info
|
|
param = "params";
|
|
UnicodeString params = testSetting
|
|
=currentCase->getString(param, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Unable to get parameter '"+param+"' "
|
|
+ UnicodeString(" - "));
|
|
continue;
|
|
}
|
|
paramsSet.parseFrom(testSetting, status); // parse with inheritance.
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Failed to parse '"+param+"' parameter: "
|
|
+testSetting);
|
|
continue;
|
|
}
|
|
|
|
toSet.clear();
|
|
// #4 'to' info
|
|
param = "to";
|
|
UnicodeString to = testSetting=currentCase->getString(param, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+": Unable to get parameter '"+param+"' "
|
|
+ UnicodeString(" - "));
|
|
continue;
|
|
}
|
|
if(to.startsWith(kMILLIS)){
|
|
UnicodeString millis = UnicodeString(to, kMILLIS.length());
|
|
useDate = TRUE;
|
|
toDate = udbg_stod(millis);
|
|
} else if(toSet.parseFrom(testSetting, &fromSet, status)<0 || U_FAILURE(status)){
|
|
errln(caseString+": Failed to parse '"+param+"' parameter: "
|
|
+testSetting);
|
|
continue;
|
|
}
|
|
|
|
UnicodeString caseContentsString = locale+": from "+from+": "
|
|
+operation +" [[[ "+params+" ]]] >>> "+to;
|
|
logln(caseString+": "+caseContentsString);
|
|
|
|
// ------
|
|
// now, do it.
|
|
|
|
/// prepare calendar
|
|
if(useDate){
|
|
fromCalendar->setTime(fromDate, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+" FAIL: Failed to set time on Source calendar: "
|
|
+ u_errorName(status));
|
|
return;
|
|
}
|
|
} else {
|
|
fromSet.setOnCalendar(fromCalendar, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+" FAIL: Failed to set on Source calendar: "
|
|
+ u_errorName(status));
|
|
return;
|
|
}
|
|
}
|
|
|
|
diffSet.clear();
|
|
// Is the calendar sane after being set?
|
|
if (!fromSet.matches(fromCalendar, diffSet, status)) {
|
|
UnicodeString diffs = diffSet.diffFrom(fromSet, status);
|
|
errln((UnicodeString)"FAIL: "+caseString
|
|
+", SET SOURCE calendar was not set: Differences: "+ diffs
|
|
+"', status: "+ u_errorName(status));
|
|
} else if (U_FAILURE(status)) {
|
|
errln("FAIL: "+caseString+" SET SOURCE calendar Failed to match: "
|
|
+u_errorName(status));
|
|
} else {
|
|
logln("PASS: "+caseString+" SET SOURCE calendar match.");
|
|
}
|
|
|
|
// to calendar - copy of from calendar
|
|
toCalendar = fromCalendar->clone();
|
|
|
|
/// perform op
|
|
for (int q=0; q<UCAL_FIELD_COUNT; q++) {
|
|
if (paramsSet.isSet((UCalendarDateFields)q)) {
|
|
if (operation == kROLL) {
|
|
toCalendar->roll((UCalendarDateFields)q,
|
|
paramsSet.get((UCalendarDateFields)q), status);
|
|
} else if (operation == kADD) {
|
|
toCalendar->add((UCalendarDateFields)q,
|
|
paramsSet.get((UCalendarDateFields)q), status);
|
|
} else {
|
|
errln(caseString+ " FAIL: unknown operation "+ operation);
|
|
}
|
|
logln(operation + " of "+ paramsSet.get((UCalendarDateFields)q)
|
|
+" -> "+u_errorName(status));
|
|
}
|
|
}
|
|
if (U_FAILURE(status)) {
|
|
errln(caseString+" FAIL: after "+operation+" of "+params+" -> "
|
|
+u_errorName(status));
|
|
continue;
|
|
}
|
|
|
|
// now - what's the result?
|
|
diffSet.clear();
|
|
|
|
if(useDate){
|
|
if(!(toCalendar->getTime(status)==toDate) || U_FAILURE(status)){
|
|
errln("FAIL: "+caseString+" Match operation had an error: "
|
|
+u_errorName(status));
|
|
}else{
|
|
logln(caseString + " SUCCESS: got=expected="+toDate);
|
|
logln("PASS: "+caseString+" matched!");
|
|
}
|
|
} else if (!toSet.matches(toCalendar, diffSet, status)) {
|
|
UnicodeString diffs = diffSet.diffFrom(toSet, status);
|
|
errln((UnicodeString)"FAIL: "+caseString+" - , "+caseContentsString
|
|
+" Differences: "+ diffs +"', status: "
|
|
+ u_errorName(status));
|
|
}else if (U_FAILURE(status)) {
|
|
errln("FAIL: "+caseString+" Match operation had an error: "
|
|
+u_errorName(status));
|
|
}else {
|
|
logln("PASS: "+caseString+" matched!");
|
|
}
|
|
|
|
delete fromCalendar;
|
|
delete toCalendar;
|
|
}
|
|
}
|
|
|
|
void DataDrivenCalendarTest::testConvert(int32_t n,
|
|
const CalendarFieldsSet &fromSet, Calendar *fromCalendar,
|
|
const CalendarFieldsSet &toSet, Calendar *toCalendar, UBool forward) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UnicodeString thisString = (UnicodeString)"#"+n+" "+(forward ? "forward"
|
|
: "reverse")+" "+fromCalendar->getType()+"->"+toCalendar->getType();
|
|
|
|
fromCalendar->clear();
|
|
|
|
fromSet.setOnCalendar(fromCalendar, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("FAIL: Failed to set on Source calendar: %s", u_errorName(status));
|
|
return;
|
|
}
|
|
|
|
CalendarFieldsSet diffSet;
|
|
|
|
diffSet.clear();
|
|
// Is the calendar sane at the first?
|
|
if (!fromSet.matches(fromCalendar, diffSet, status)) {
|
|
UnicodeString diffs = diffSet.diffFrom(fromSet, status);
|
|
errln((UnicodeString)"FAIL: "+thisString
|
|
+", SOURCE calendar was not set: Differences: "+ diffs
|
|
+"', status: "+ u_errorName(status));
|
|
} else if (U_FAILURE(status)) {
|
|
errln("FAIL: "+thisString+" SOURCE calendar Failed to match: "
|
|
+u_errorName(status));
|
|
} else {
|
|
logln("PASS: "+thisString+" SOURCE calendar match.");
|
|
}
|
|
|
|
//logln("Set Source calendar: " + from);
|
|
|
|
UDate fromTime = fromCalendar->getTime(status);
|
|
if (U_FAILURE(status)) {
|
|
errln("FAIL: Failed to get Source time: %s", u_errorName(status));
|
|
return;
|
|
}
|
|
|
|
diffSet.clear();
|
|
// Is the calendar sane after being set?
|
|
if (!fromSet.matches(fromCalendar, diffSet, status)) {
|
|
UnicodeString diffs = diffSet.diffFrom(fromSet, status);
|
|
errln((UnicodeString)"FAIL: "+thisString
|
|
+", SET SOURCE calendar was not set: Differences: "+ diffs
|
|
+"', status: "+ u_errorName(status));
|
|
} else if (U_FAILURE(status)) {
|
|
errln("FAIL: "+thisString+" SET SOURCE calendar Failed to match: "
|
|
+u_errorName(status));
|
|
} else {
|
|
logln("PASS: "+thisString+" SET SOURCE calendar match.");
|
|
}
|
|
|
|
toCalendar->clear();
|
|
toCalendar->setTime(fromTime, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("FAIL: Failed to set Target time: %s", u_errorName(status));
|
|
return;
|
|
}
|
|
|
|
diffSet.clear();
|
|
if (!toSet.matches(toCalendar, diffSet, status)) {
|
|
UnicodeString diffs = diffSet.diffFrom(toSet, status);
|
|
errln((UnicodeString)"FAIL: "+thisString+", Differences: "+ diffs
|
|
+"', status: "+ u_errorName(status));
|
|
SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy G"), status);
|
|
UnicodeString fromString;
|
|
fmt.format(fromTime, fromString);
|
|
logln("Source Time: "+fromString+", Source Calendar: "
|
|
+fromCalendar->getType());
|
|
} else if (U_FAILURE(status)) {
|
|
errln("FAIL: "+thisString+" Failed to match: "+u_errorName(status));
|
|
} else {
|
|
logln("PASS: "+thisString+" match.");
|
|
}
|
|
}
|
|
|
|
void DataDrivenCalendarTest::testConvert(TestData *testData,
|
|
const DataMap *settings, UBool forward) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> toCalendar;
|
|
const DataMap *currentCase= NULL;
|
|
char toCalLoc[256] = "";
|
|
char fromCalLoc[256] = "";
|
|
// build to calendar
|
|
UnicodeString testSetting = settings->getString("ToCalendar", status);
|
|
if (U_SUCCESS(status)) {
|
|
testSetting.extract(0, testSetting.length(), toCalLoc, (const char*)0);
|
|
toCalendar.adoptInstead(Calendar::createInstance(toCalLoc, status));
|
|
if (U_FAILURE(status)) {
|
|
dataerrln(UnicodeString("Unable to instantiate ToCalendar for ")+testSetting);
|
|
return;
|
|
}
|
|
}
|
|
|
|
CalendarFieldsSet fromSet, toSet, diffSet;
|
|
SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy / YYYY'-W'ww-ee"),
|
|
status);
|
|
if (U_FAILURE(status)) {
|
|
errcheckln(status, "FAIL: Couldn't create SimpleDateFormat: %s",
|
|
u_errorName(status));
|
|
return;
|
|
}
|
|
// Start the processing
|
|
int n = 0;
|
|
while (testData->nextCase(currentCase, status)) {
|
|
++n;
|
|
LocalPointer<Calendar> fromCalendar;
|
|
UnicodeString locale = currentCase->getString("locale", status);
|
|
if (U_SUCCESS(status)) {
|
|
locale.extract(0, locale.length(), fromCalLoc, (const char*)0); // default codepage. Invariant codepage doesn't have '@'!
|
|
fromCalendar.adoptInstead(Calendar::createInstance(fromCalLoc, status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Unable to instantiate fromCalendar for "+locale);
|
|
return;
|
|
}
|
|
} else {
|
|
errln("No 'locale' line.");
|
|
continue;
|
|
}
|
|
|
|
fromSet.clear();
|
|
toSet.clear();
|
|
|
|
UnicodeString from = currentCase->getString("from", status);
|
|
if (U_FAILURE(status)) {
|
|
errln("No 'from' line.");
|
|
continue;
|
|
}
|
|
fromSet.parseFrom(from, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Failed to parse 'from' parameter: "+from);
|
|
continue;
|
|
}
|
|
UnicodeString to = currentCase->getString("to", status);
|
|
if (U_FAILURE(status)) {
|
|
errln("No 'to' line.");
|
|
continue;
|
|
}
|
|
toSet.parseFrom(to, &fromSet, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Failed to parse 'to' parameter: "+to);
|
|
continue;
|
|
}
|
|
|
|
// now, do it.
|
|
if (forward) {
|
|
logln((UnicodeString)"#"+n+" "+locale+"/"+from+" >>> "+toCalLoc+"/"
|
|
+to);
|
|
testConvert(n, fromSet, fromCalendar.getAlias(), toSet, toCalendar.getAlias(), forward);
|
|
} else {
|
|
logln((UnicodeString)"#"+n+" "+locale+"/"+from+" <<< "+toCalLoc+"/"
|
|
+to);
|
|
testConvert(n, toSet, toCalendar.getAlias(), fromSet, fromCalendar.getAlias(), forward);
|
|
}
|
|
}
|
|
}
|
|
|
|
void DataDrivenCalendarTest::processTest(TestData *testData) {
|
|
//Calendar *cal= NULL;
|
|
//const UChar *arguments= NULL;
|
|
//int32_t argLen = 0;
|
|
char testType[256] = "";
|
|
const DataMap *settings= NULL;
|
|
//const UChar *type= NULL;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UnicodeString testSetting;
|
|
int n = 0;
|
|
while (testData->nextSettings(settings, status)) {
|
|
status = U_ZERO_ERROR;
|
|
// try to get a locale
|
|
testSetting = settings->getString("Type", status);
|
|
if (U_SUCCESS(status)) {
|
|
if ((++n)>0) {
|
|
logln("---");
|
|
}
|
|
logln(testSetting + "---");
|
|
testSetting.extract(0, testSetting.length(), testType, "");
|
|
} else {
|
|
errln("Unable to extract 'Type'. Skipping..");
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(testType, "convert_fwd")) {
|
|
testConvert(testData, settings, true);
|
|
} else if (!strcmp(testType, "convert_rev")) {
|
|
testConvert(testData, settings, false);
|
|
} else if (!strcmp(testType, "ops")) {
|
|
testOps(testData, settings);
|
|
} else {
|
|
errln("Unknown type: %s", testType);
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|