// 2001-09-12 Benjamin Kosnik // Copyright (C) 2001, 2002 Free Software Foundation // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 2, 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 General Public License for more details. // You should have received a copy of the GNU General Public License along // with this library; see the file COPYING. If not, write to the Free // Software Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, // USA. // 22.2.6.1.1 money_get members #include #include #include // XXX This test is not working for non-glibc locale models. // { dg-do run { xfail *-*-* } } // test string version void test01() { using namespace std; typedef money_base::part part; typedef money_base::pattern pattern; typedef istreambuf_iterator iterator_type; bool test = true; // basic construction locale loc_c = locale::classic(); locale loc_hk("en_HK"); locale loc_fr("fr_FR@euro"); locale loc_de("de_DE"); VERIFY( loc_c != loc_de ); VERIFY( loc_hk != loc_fr ); VERIFY( loc_hk != loc_de ); VERIFY( loc_de != loc_fr ); // cache the moneypunct facets typedef moneypunct __money_true; typedef moneypunct __money_false; const __money_true& monpunct_c_t = use_facet<__money_true>(loc_c); const __money_true& monpunct_de_t = use_facet<__money_true>(loc_de); const __money_false& monpunct_c_f = use_facet<__money_false>(loc_c); const __money_false& monpunct_de_f = use_facet<__money_false>(loc_de); const __money_true& monpunct_hk_t = use_facet<__money_true>(loc_hk); const __money_false& monpunct_hk_f = use_facet<__money_false>(loc_hk); // sanity check the data is correct. const string empty; // total EPA budget FY 2002 const string digits1("720000000000"); // est. cost, national missile "defense", expressed as a loss in USD 2001 const string digits2("-10000000000000"); // not valid input const string digits3("-A"); // input less than frac_digits const string digits4("-1"); iterator_type end; istringstream iss; iss.imbue(loc_de); // cache the money_get facet const money_get& mon_get = use_facet >(iss.getloc()); iss.str("7.200.000.000,00 "); iterator_type is_it01(iss); string result1; ios_base::iostate err01 = ios_base::goodbit; mon_get.get(is_it01, end, true, iss, err01, result1); VERIFY( result1 == digits1 ); VERIFY( err01 == ios_base::eofbit ); iss.str("7.200.000.000,00 "); iterator_type is_it02(iss); string result2; ios_base::iostate err02 = ios_base::goodbit; mon_get.get(is_it02, end, true, iss, err02, result2); VERIFY( result2 == digits1 ); VERIFY( err02 == ios_base::eofbit ); iss.str("7.200.000.000,00 a"); iterator_type is_it03(iss); string result3; ios_base::iostate err03 = ios_base::goodbit; mon_get.get(is_it03, end, true, iss, err03, result3); VERIFY( result3 == digits1 ); VERIFY( err03 == ios_base::goodbit ); iss.str(""); iterator_type is_it04(iss); string result4; ios_base::iostate err04 = ios_base::goodbit; mon_get.get(is_it04, end, true, iss, err04, result4); VERIFY( result4 == empty ); VERIFY( err04 == ios_base::failbit | ios_base::eofbit ); iss.str("working for enlightenment and peace in a mad world"); iterator_type is_it05(iss); string result5; ios_base::iostate err05 = ios_base::goodbit; mon_get.get(is_it05, end, true, iss, err05, result5); VERIFY( result5 == empty ); VERIFY( err05 == ios_base::failbit ); // now try with showbase, to get currency symbol in format iss.setf(ios_base::showbase); iss.str("7.200.000.000,00 DEM "); iterator_type is_it06(iss); string result6; ios_base::iostate err06 = ios_base::goodbit; mon_get.get(is_it06, end, true, iss, err06, result6); VERIFY( result6 == digits1 ); VERIFY( err06 == ios_base::eofbit ); iss.str("7.200.000.000,00 DEM "); // Extra space. iterator_type is_it07(iss); string result7; ios_base::iostate err07 = ios_base::goodbit; mon_get.get(is_it07, end, true, iss, err07, result7); VERIFY( result7 == digits1 ); VERIFY( err07 == ios_base::goodbit ); iss.str("7.200.000.000,00 DM"); iterator_type is_it08(iss); string result8; ios_base::iostate err08 = ios_base::goodbit; mon_get.get(is_it08, end, false, iss, err08, result8); VERIFY( result8 == digits1 ); VERIFY( err08 == ios_base::eofbit ); iss.imbue(loc_hk); iss.str("HK$7,200,000,000.00"); iterator_type is_it09(iss); string result9; ios_base::iostate err09 = ios_base::goodbit; mon_get.get(is_it09, end, false, iss, err09, result9); VERIFY( result9 == digits1 ); VERIFY( err09 == ios_base::eofbit ); iss.str("(HKD 100,000,000,000.00)"); iterator_type is_it10(iss); string result10; ios_base::iostate err10 = ios_base::goodbit; mon_get.get(is_it10, end, true, iss, err10, result10); VERIFY( result10 == digits2 ); VERIFY( err10 == ios_base::goodbit ); iss.str("(HKD .01)"); iterator_type is_it11(iss); string result11; ios_base::iostate err11 = ios_base::goodbit; mon_get.get(is_it11, end, true, iss, err11, result11); VERIFY( result11 == digits4 ); VERIFY( err11 == ios_base::goodbit ); // for the "en_HK" locale the parsing of the very same input streams must // be successful without showbase too, since the symbol field appears in // the first positions in the format and the symbol, when present, must be // consumed. iss.unsetf(ios_base::showbase); iss.str("HK$7,200,000,000.00"); iterator_type is_it12(iss); string result12; ios_base::iostate err12 = ios_base::goodbit; mon_get.get(is_it12, end, false, iss, err12, result12); VERIFY( result12 == digits1 ); VERIFY( err12 == ios_base::eofbit ); iss.str("(HKD 100,000,000,000.00)"); iterator_type is_it13(iss); string result13; ios_base::iostate err13 = ios_base::goodbit; mon_get.get(is_it13, end, true, iss, err13, result13); VERIFY( result13 == digits2 ); VERIFY( err13 == ios_base::goodbit ); iss.str("(HKD .01)"); iterator_type is_it14(iss); string result14; ios_base::iostate err14 = ios_base::goodbit; mon_get.get(is_it14, end, true, iss, err14, result14); VERIFY( result14 == digits4 ); VERIFY( err14 == ios_base::goodbit ); } // test double version void test02() { using namespace std; typedef money_base::part part; typedef money_base::pattern pattern; typedef istreambuf_iterator iterator_type; bool test = true; // basic construction locale loc_c = locale::classic(); locale loc_hk("en_HK"); locale loc_fr("fr_FR@euro"); locale loc_de("de_DE"); VERIFY( loc_c != loc_de ); VERIFY( loc_hk != loc_fr ); VERIFY( loc_hk != loc_de ); VERIFY( loc_de != loc_fr ); // cache the moneypunct facets typedef moneypunct __money_true; typedef moneypunct __money_false; const __money_true& monpunct_c_t = use_facet<__money_true>(loc_c); const __money_true& monpunct_de_t = use_facet<__money_true>(loc_de); const __money_false& monpunct_c_f = use_facet<__money_false>(loc_c); const __money_false& monpunct_de_f = use_facet<__money_false>(loc_de); const __money_true& monpunct_hk_t = use_facet<__money_true>(loc_hk); const __money_false& monpunct_hk_f = use_facet<__money_false>(loc_hk); // sanity check the data is correct. const string empty; // total EPA budget FY 2002 const long double digits1 = 720000000000; // est. cost, national missile "defense", expressed as a loss in USD 2001 const long double digits2 = -10000000000000; // input less than frac_digits const long double digits4 = -1; iterator_type end; istringstream iss; iss.imbue(loc_de); // cache the money_get facet const money_get& mon_get = use_facet >(iss.getloc()); iss.str("7.200.000.000,00 "); iterator_type is_it01(iss); long double result1; ios_base::iostate err01 = ios_base::goodbit; mon_get.get(is_it01, end, true, iss, err01, result1); VERIFY( result1 == digits1 ); VERIFY( err01 == ios_base::eofbit ); iss.str("7.200.000.000,00 "); iterator_type is_it02(iss); long double result2; ios_base::iostate err02 = ios_base::goodbit; mon_get.get(is_it02, end, false, iss, err02, result2); VERIFY( result2 == digits1 ); VERIFY( err02 == ios_base::eofbit ); // now try with showbase, to get currency symbol in format iss.setf(ios_base::showbase); iss.imbue(loc_hk); iss.str("(HKD .01)"); iterator_type is_it03(iss); long double result3; ios_base::iostate err03 = ios_base::goodbit; mon_get.get(is_it03, end, true, iss, err03, result3); VERIFY( result3 == digits4 ); VERIFY( err03 == ios_base::goodbit ); } void test03() { using namespace std; bool test = true; // Check money_get works with other iterators besides streambuf // input iterators. typedef string::const_iterator iter_type; typedef money_get mon_get_type; const ios_base::iostate goodbit = ios_base::goodbit; const ios_base::iostate eofbit = ios_base::eofbit; ios_base::iostate err = goodbit; const locale loc_c = locale::classic(); const string str = "0.01Eleanor Roosevelt"; istringstream iss; iss.imbue(locale(loc_c, new mon_get_type)); // Iterator advanced, state, output. const mon_get_type& mg = use_facet(iss.getloc()); // 01 string string res1; iter_type end1 = mg.get(str.begin(), str.end(), false, iss, err, res1); string rem1(end1, str.end()); VERIFY( err == goodbit ); VERIFY( res1 == "1" ); VERIFY( rem1 == "Eleanor Roosevelt" ); // 02 long double iss.clear(); err = goodbit; long double res2; iter_type end2 = mg.get(str.begin(), str.end(), false, iss, err, res2); string rem2(end2, str.end()); VERIFY( err == goodbit ); VERIFY( res2 == 1 ); VERIFY( rem2 == "Eleanor Roosevelt" ); } // libstdc++/5280 void test04() { #ifdef _GLIBCPP_HAVE_SETENV // Set the global locale to non-"C". std::locale loc_de("de_DE"); std::locale::global(loc_de); // Set LANG environment variable to de_DE. const char* oldLANG = getenv("LANG"); if (!setenv("LANG", "de_DE", 1)) { test01(); test02(); test03(); setenv("LANG", oldLANG ? oldLANG : "", 1); } #endif } struct My_money_io : public std::moneypunct { char_type do_decimal_point() const { return '.'; } std::string do_grouping() const { return "\004"; } std::string do_curr_symbol() const { return "$"; } std::string do_positive_sign() const { return ""; } std::string do_negative_sign() const { return "-"; } int do_frac_digits() const { return 2; } pattern do_pos_format() const { pattern pat = { { symbol, none, sign, value } }; return pat; } pattern do_neg_format() const { pattern pat = { { symbol, none, sign, value } }; return pat; } }; // libstdc++/5579 void test05() { using namespace std; typedef istreambuf_iterator InIt; bool test = true; locale loc(locale::classic(), new My_money_io); string bufferp("$1234.56"); string buffern("$-1234.56"); string bufferp_ns("1234.56"); string buffern_ns("-1234.56"); bool intl = false; InIt iendp, iendn, iendp_ns, iendn_ns; ios_base::iostate err; string valp, valn, valp_ns, valn_ns; const money_get& mg = use_facet >(loc); istringstream fmtp(bufferp); fmtp.imbue(loc); InIt ibegp(fmtp); mg.get(ibegp,iendp,intl,fmtp,err,valp); VERIFY( valp == "123456" ); istringstream fmtn(buffern); fmtn.imbue(loc); InIt ibegn(fmtn); mg.get(ibegn,iendn,intl,fmtn,err,valn); VERIFY( valn == "-123456" ); istringstream fmtp_ns(bufferp_ns); fmtp_ns.imbue(loc); InIt ibegp_ns(fmtp_ns); mg.get(ibegp_ns,iendp_ns,intl,fmtp_ns,err,valp_ns); VERIFY( valp_ns == "123456" ); istringstream fmtn_ns(buffern_ns); fmtn_ns.imbue(loc); InIt ibegn_ns(fmtn_ns); mg.get(ibegn_ns,iendn_ns,intl,fmtn_ns,err,valn_ns); VERIFY( valn_ns == "-123456" ); } // We were appending to the string val passed by reference, instead // of constructing a temporary candidate, eventually copied into // val in case of successful parsing. void test06() { using namespace std; bool test = true; typedef istreambuf_iterator InIt; InIt iend1, iend2, iend3; locale loc; string buffer1("123"); string buffer2("456"); string buffer3("Golgafrincham"); // From Nathan's original idea. string val; ios_base::iostate err; const money_get& mg = use_facet >(loc); istringstream fmt1(buffer1); InIt ibeg1(fmt1); mg.get(ibeg1,iend1,false,fmt1,err,val); VERIFY( val == buffer1 ); istringstream fmt2(buffer2); InIt ibeg2(fmt2); mg.get(ibeg2,iend2,false,fmt2,err,val); VERIFY( val == buffer2 ); val = buffer3; istringstream fmt3(buffer3); InIt ibeg3(fmt3); mg.get(ibeg3,iend3,false,fmt3,err,val); VERIFY( val == buffer3 ); } struct My_money_io_a : public std::moneypunct { char_type do_decimal_point() const { return '.'; } std::string do_grouping() const { return "\004"; } std::string do_curr_symbol() const { return "$"; } std::string do_positive_sign() const { return "()"; } int do_frac_digits() const { return 2; } pattern do_pos_format() const { pattern pat = { { sign, value, space, symbol } }; return pat; } }; struct My_money_io_b : public std::moneypunct { char_type do_decimal_point() const { return '.'; } std::string do_grouping() const { return "\004"; } std::string do_curr_symbol() const { return "$"; } std::string do_positive_sign() const { return "()"; } int do_frac_digits() const { return 2; } pattern do_pos_format() const { pattern pat = { { sign, value, symbol, none } }; return pat; } }; // This one exercises patterns of the type { X, Y, Z, symbol } and // { X, Y, symbol, none } for a two character long sign. Therefore // the optional symbol (showbase is false by default) must be consumed // if present, since "rest of the sign" is left to read. void test07() { using namespace std; typedef istreambuf_iterator InIt; bool intl = false; bool test = true; ios_base::iostate err; locale loc_a(locale::classic(), new My_money_io_a); string buffer_a("(1234.56 $)"); string buffer_a_ns("(1234.56 )"); InIt iend_a, iend_a_ns; string val_a, val_a_ns; const money_get& mg_a = use_facet >(loc_a); istringstream fmt_a(buffer_a); fmt_a.imbue(loc_a); InIt ibeg_a(fmt_a); mg_a.get(ibeg_a,iend_a,intl,fmt_a,err,val_a); VERIFY( val_a == "123456" ); istringstream fmt_a_ns(buffer_a_ns); fmt_a_ns.imbue(loc_a); InIt ibeg_a_ns(fmt_a_ns); mg_a.get(ibeg_a_ns,iend_a_ns,intl,fmt_a_ns,err,val_a_ns); VERIFY( val_a_ns == "123456" ); locale loc_b(locale::classic(), new My_money_io_b); string buffer_b("(1234.56$)"); string buffer_b_ns("(1234.56)"); InIt iend_b, iend_b_ns; string val_b, val_b_ns; const money_get& mg_b = use_facet >(loc_b); istringstream fmt_b(buffer_b); fmt_b.imbue(loc_b); InIt ibeg_b(fmt_b); mg_b.get(ibeg_b,iend_b,intl,fmt_b,err,val_b); VERIFY( val_b == "123456" ); istringstream fmt_b_ns(buffer_b_ns); fmt_b_ns.imbue(loc_b); InIt ibeg_b_ns(fmt_b_ns); mg_b.get(ibeg_b_ns,iend_b_ns,intl,fmt_b_ns,err,val_b_ns); VERIFY( val_b_ns == "123456" ); } int main() { test01(); test02(); test03(); test04(); test05(); test06(); test07(); return 0; }