mirror of
https://github.com/unicode-org/icu.git
synced 2025-04-06 22:15:31 +00:00
Merge pull request #20 from icu-units/units_constants
Test that addSingleFactorConstant knowns all unitConstants.
This commit is contained in:
commit
6c0c3bb640
3 changed files with 76 additions and 17 deletions
|
@ -22,7 +22,7 @@
|
|||
U_NAMESPACE_BEGIN
|
||||
namespace units {
|
||||
|
||||
void Factor::multiplyBy(const Factor &rhs) {
|
||||
void U_I18N_API Factor::multiplyBy(const Factor &rhs) {
|
||||
factorNum *= rhs.factorNum;
|
||||
factorDen *= rhs.factorDen;
|
||||
for (int i = 0; i < CONSTANTS_COUNT; i++) {
|
||||
|
@ -35,7 +35,7 @@ void Factor::multiplyBy(const Factor &rhs) {
|
|||
offset = std::max(rhs.offset, offset);
|
||||
}
|
||||
|
||||
void Factor::divideBy(const Factor &rhs) {
|
||||
void U_I18N_API Factor::divideBy(const Factor &rhs) {
|
||||
factorNum *= rhs.factorDen;
|
||||
factorDen *= rhs.factorNum;
|
||||
for (int i = 0; i < CONSTANTS_COUNT; i++) {
|
||||
|
@ -48,7 +48,7 @@ void Factor::divideBy(const Factor &rhs) {
|
|||
offset = std::max(rhs.offset, offset);
|
||||
}
|
||||
|
||||
void Factor::power(int32_t power) {
|
||||
void U_I18N_API Factor::power(int32_t power) {
|
||||
// multiply all the constant by the power.
|
||||
for (int i = 0; i < CONSTANTS_COUNT; i++) {
|
||||
constants[i] *= power;
|
||||
|
@ -66,7 +66,7 @@ void Factor::power(int32_t power) {
|
|||
}
|
||||
}
|
||||
|
||||
void Factor::flip() {
|
||||
void U_I18N_API Factor::flip() {
|
||||
std::swap(factorNum, factorDen);
|
||||
|
||||
for (int i = 0; i < CONSTANTS_COUNT; i++) {
|
||||
|
@ -74,7 +74,7 @@ void Factor::flip() {
|
|||
}
|
||||
}
|
||||
|
||||
void Factor::applySiPrefix(UMeasureSIPrefix siPrefix) {
|
||||
void U_I18N_API Factor::applySiPrefix(UMeasureSIPrefix siPrefix) {
|
||||
if (siPrefix == UMeasureSIPrefix::UMEASURE_SI_PREFIX_ONE) return; // No need to do anything
|
||||
|
||||
double siApplied = std::pow(10.0, std::abs(siPrefix));
|
||||
|
@ -87,16 +87,20 @@ void Factor::applySiPrefix(UMeasureSIPrefix siPrefix) {
|
|||
factorNum *= siApplied;
|
||||
}
|
||||
|
||||
void Factor::substituteConstants() {
|
||||
double constantsValues[CONSTANTS_COUNT];
|
||||
|
||||
// TODO: Load those constant values from units data.
|
||||
constantsValues[CONSTANT_FT2M] = 0.3048;
|
||||
constantsValues[CONSTANT_PI] = 411557987.0 / 131002976.0;
|
||||
constantsValues[CONSTANT_GRAVITY] = 9.80665;
|
||||
constantsValues[CONSTANT_G] = 6.67408E-11;
|
||||
constantsValues[CONSTANT_LB2KG] = 0.45359237;
|
||||
constantsValues[CONSTANT_GAL_IMP2M3] = 0.00454609;
|
||||
void U_I18N_API Factor::substituteConstants() {
|
||||
// These values are a hard-coded subset of unitConstants in the units
|
||||
// resources file. A unit test checks that all constants in the resource
|
||||
// file are at least recognised by the code. Derived constants' values or
|
||||
// hard-coded derivations are not checked.
|
||||
// double constantsValues[CONSTANTS_COUNT];
|
||||
static const double constantsValues[CONSTANTS_COUNT] = {
|
||||
[CONSTANT_FT2M] = 0.3048, //
|
||||
[CONSTANT_PI] = 411557987.0 / 131002976.0, //
|
||||
[CONSTANT_GRAVITY] = 9.80665, //
|
||||
[CONSTANT_G] = 6.67408E-11, //
|
||||
[CONSTANT_GAL_IMP2M3] = 0.00454609, //
|
||||
[CONSTANT_LB2KG] = 0.45359237, //
|
||||
};
|
||||
|
||||
for (int i = 0; i < CONSTANTS_COUNT; i++) {
|
||||
if (this->constants[i] == 0) {
|
||||
|
@ -322,9 +326,12 @@ void loadConversionRate(ConversionRate &conversionRate, const MeasureUnit &sourc
|
|||
|
||||
} // namespace
|
||||
|
||||
// Conceptually, this modifies factor: factor *= baseStr^(signum*power).
|
||||
//
|
||||
// baseStr must be a known constant or a value that strToDouble() is able to
|
||||
// parse.
|
||||
void U_I18N_API addSingleFactorConstant(StringPiece baseStr, int32_t power, Signum signum,
|
||||
Factor &factor, UErrorCode &status) {
|
||||
|
||||
if (baseStr == "ft_to_m") {
|
||||
factor.constants[CONSTANT_FT2M] += power * signum;
|
||||
} else if (baseStr == "ft2_to_m2") {
|
||||
|
|
|
@ -35,7 +35,7 @@ typedef enum Signum {
|
|||
} Signum;
|
||||
|
||||
/* Represents a conversion factor */
|
||||
struct Factor {
|
||||
struct U_I18N_API Factor {
|
||||
double factorNum = 1;
|
||||
double factorDen = 1;
|
||||
double offset = 0;
|
||||
|
|
|
@ -18,10 +18,12 @@
|
|||
#include "unicode/measunit.h"
|
||||
#include "unicode/unistr.h"
|
||||
#include "unicode/unum.h"
|
||||
#include "unicode/ures.h"
|
||||
#include "unitconverter.h"
|
||||
#include "unitsdata.h"
|
||||
#include "unitsrouter.h"
|
||||
#include "uparse.h"
|
||||
#include "uresimp.h"
|
||||
|
||||
struct UnitConversionTestCase {
|
||||
const StringPiece source;
|
||||
|
@ -39,6 +41,7 @@ class UnitsTest : public IntlTest {
|
|||
|
||||
void runIndexedTest(int32_t index, UBool exec, const char *&name, char *par = NULL);
|
||||
|
||||
void testUnitConstantFreshness();
|
||||
void testConversionCapability();
|
||||
void testConversions();
|
||||
void testPreferences();
|
||||
|
@ -55,6 +58,7 @@ void UnitsTest::runIndexedTest(int32_t index, UBool exec, const char *&name, cha
|
|||
logln("TestSuite UnitsTest: ");
|
||||
}
|
||||
TESTCASE_AUTO_BEGIN;
|
||||
TESTCASE_AUTO(testUnitConstantFreshness);
|
||||
TESTCASE_AUTO(testConversionCapability);
|
||||
TESTCASE_AUTO(testConversions);
|
||||
TESTCASE_AUTO(testPreferences);
|
||||
|
@ -65,6 +69,54 @@ void UnitsTest::runIndexedTest(int32_t index, UBool exec, const char *&name, cha
|
|||
TESTCASE_AUTO_END;
|
||||
}
|
||||
|
||||
// Tests the hard-coded constants in the code against constants that appear in
|
||||
// units.txt.
|
||||
void UnitsTest::testUnitConstantFreshness() {
|
||||
IcuTestErrorCode status(*this, "testUnitConstantFreshness");
|
||||
LocalUResourceBundlePointer unitsBundle(ures_openDirect(NULL, "units", status));
|
||||
LocalUResourceBundlePointer unitConstants(
|
||||
ures_getByKey(unitsBundle.getAlias(), "unitConstants", NULL, status));
|
||||
|
||||
while (ures_hasNext(unitConstants.getAlias())) {
|
||||
int32_t len;
|
||||
const char *constant = NULL;
|
||||
ures_getNextString(unitConstants.getAlias(), &len, &constant, status);
|
||||
|
||||
Factor factor;
|
||||
addSingleFactorConstant(constant, 1, POSITIVE, factor, status);
|
||||
if (status.errDataIfFailureAndReset(
|
||||
"addSingleFactorConstant(<%s>, ...).\n\n"
|
||||
"If U_INVALID_FORMAT_ERROR, please check that \"icu4c/source/i18n/unitconverter.cpp\" "
|
||||
"has all constants? Is \"%s\" a new constant?\n",
|
||||
constant, constant)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check the values of constants that have a simple numeric value
|
||||
factor.substituteConstants();
|
||||
int32_t uLen;
|
||||
UnicodeString uVal = ures_getStringByKey(unitConstants.getAlias(), constant, &uLen, status);
|
||||
CharString val;
|
||||
val.appendInvariantChars(uVal, status);
|
||||
if (status.errDataIfFailureAndReset("Failed to get constant value for %s.", constant)) {
|
||||
continue;
|
||||
}
|
||||
DecimalQuantity dqVal;
|
||||
UErrorCode parseStatus = U_ZERO_ERROR;
|
||||
// TODO(units): unify with strToDouble() in unitconverter.cpp
|
||||
dqVal.setToDecNumber(val.toStringPiece(), parseStatus);
|
||||
if (!U_SUCCESS(parseStatus)) {
|
||||
// Not simple to parse, skip validating this constant's value. (We
|
||||
// leave catching mistakes to the data-driven integration tests.)
|
||||
continue;
|
||||
}
|
||||
double expectedNumerator = dqVal.toDouble();
|
||||
assertEquals(UnicodeString("Constant ") + constant + u" numerator", expectedNumerator,
|
||||
factor.factorNum);
|
||||
assertEquals(UnicodeString("Constant ") + constant + u" denominator", 1.0, factor.factorDen);
|
||||
}
|
||||
}
|
||||
|
||||
void UnitsTest::testConversionCapability() {
|
||||
struct TestCase {
|
||||
const char *const source;
|
||||
|
|
Loading…
Add table
Reference in a new issue