diff options
author | John Wiegley <johnw@newartisans.com> | 2009-02-01 18:35:44 -0400 |
---|---|---|
committer | John Wiegley <johnw@newartisans.com> | 2009-02-01 18:35:44 -0400 |
commit | 66b097bda722a9f6582043f806da71f2af8deaa3 (patch) | |
tree | f6fb15e32cc97360e7040b5b8b50d573a331cb1d /src | |
parent | 59a080cdb6bafa6ab20cb20c1bf1f007ea64de84 (diff) | |
download | fork-ledger-66b097bda722a9f6582043f806da71f2af8deaa3.tar.gz fork-ledger-66b097bda722a9f6582043f806da71f2af8deaa3.tar.bz2 fork-ledger-66b097bda722a9f6582043f806da71f2af8deaa3.zip |
Moved the args_to_predicate function to predicate.cc.
Diffstat (limited to 'src')
-rw-r--r-- | src/predicate.cc | 189 | ||||
-rw-r--r-- | src/predicate.h | 3 | ||||
-rw-r--r-- | src/report.cc | 124 |
3 files changed, 194 insertions, 122 deletions
diff --git a/src/predicate.cc b/src/predicate.cc new file mode 100644 index 00000000..19019ec7 --- /dev/null +++ b/src/predicate.cc @@ -0,0 +1,189 @@ +/* + * Copyright (c) 2003-2009, John Wiegley. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are + * met: + * + * - Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * - Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * - Neither the name of New Artisans LLC nor the names of its + * contributors may be used to endorse or promote products derived from + * this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include "predicate.h" + +namespace ledger { + +string args_to_predicate_expr(value_t::sequence_t::const_iterator begin, + value_t::sequence_t::const_iterator end) +{ + std::ostringstream expr; + bool append_and = false; + bool only_parenthesis; + + while (begin != end) { + string arg = (*begin).as_string(); + bool parse_argument = true; + + if (arg == "not" || arg == "NOT") { + expr << " ! "; + parse_argument = false; + append_and = false; + } + else if (arg == "and" || arg == "AND") { + expr << " & "; + parse_argument = false; + append_and = false; + } + else if (arg == "or" || arg == "OR") { + expr << " | "; + parse_argument = false; + append_and = false; + } + else if (append_and) { + if (! only_parenthesis) + expr << " & "; + } + else { + append_and = true; + } + + if (arg == "desc" || arg == "DESC" || + arg == "payee" || arg == "PAYEE") { + arg = string("@") + (*++begin).as_string(); + } + else if (arg == "note" || arg == "NOTE") { + arg = string("&") + (*++begin).as_string(); + } + else if (arg == "tag" || arg == "TAG" || + arg == "meta" || arg == "META" || + arg == "data" || arg == "DATA") { + arg = string("%") + (*++begin).as_string(); + } + else if (arg == "expr" || arg == "EXPR") { + arg = string("=") + (*++begin).as_string(); + } + + if (parse_argument) { + bool in_prefix = true; + bool in_suffix = false; + bool found_specifier = false; + bool saw_tag_char = false; + bool no_final_slash = false; + + only_parenthesis = true; + + for (const char * c = arg.c_str(); *c != '\0'; c++) { + bool consumed = false; + + if (*c != '(' && *c != ')') + only_parenthesis = false; + + if (in_prefix) { + switch (*c) { + case '(': + break; + case '@': + expr << "(payee =~ /"; + found_specifier = true; + consumed = true; + break; + case '=': + expr << "("; + found_specifier = true; + no_final_slash = true; + consumed = true; + break; + case '&': + expr << "(note =~ /"; + found_specifier = true; + consumed = true; + break; + case '%': { + bool found_metadata = false; + for (const char *q = c; *q != '\0'; q++) + if (*q == '=') { + expr << "(metadata(\"" + << string(c + 1, q - c - 1) << "\") =~ /"; + found_metadata = true; + c = q; + break; + } + if (! found_metadata) { + expr << "(tag =~ /:"; + saw_tag_char = true; + } + found_specifier = true; + consumed = true; + break; + } + case '/': + case '_': + default: + if (! found_specifier) { + expr << "(account =~ /"; + found_specifier = true; + } + in_prefix = false; + break; + } + } else { + switch (*c) { + case ')': + if (! in_suffix) { + if (found_specifier) { + if (saw_tag_char) + expr << ':'; + expr << "/)"; + } + in_suffix = true; + } + break; + default: + if (in_suffix) + throw_(parse_error, "Invalid text in specification argument"); + break; + } + } + + if (! consumed) + expr << *c; + } + + if (! in_suffix) { + if (found_specifier) { + if (saw_tag_char) + expr << ':'; + if (! no_final_slash) + expr << "/"; + expr << ")"; + } + } + } + + begin++; + } + + return expr.str(); +} + +} // namespace ledger diff --git a/src/predicate.h b/src/predicate.h index c25ff915..5e2057ca 100644 --- a/src/predicate.h +++ b/src/predicate.h @@ -83,6 +83,9 @@ public: } }; +string args_to_predicate_expr(value_t::sequence_t::const_iterator begin, + value_t::sequence_t::const_iterator end); + } // namespace ledger #endif // _PREDICATE_H diff --git a/src/report.cc b/src/report.cc index d0760a03..a9f4f96b 100644 --- a/src/report.cc +++ b/src/report.cc @@ -135,126 +135,6 @@ namespace { return string_value(out.str()); } - string args_to_predicate(value_t::sequence_t::const_iterator begin, - value_t::sequence_t::const_iterator end) - { - std::ostringstream expr; - bool append_and = false; - bool only_parenthesis; - - while (begin != end) { - const string& arg((*begin).as_string()); - const char * p = arg.c_str(); - - bool parse_argument = true; - - if (arg == "not" || arg == "NOT") { - expr << " ! "; - parse_argument = false; - append_and = false; - } - else if (arg == "and" || arg == "AND") { - expr << " & "; - parse_argument = false; - append_and = false; - } - else if (arg == "or" || arg == "OR") { - expr << " | "; - parse_argument = false; - append_and = false; - } - else if (append_and) { - if (! only_parenthesis) - expr << " & "; - } - else { - append_and = true; - } - - if (parse_argument) { - bool in_prefix = true; - bool in_suffix = false; - bool found_specifier = false; - bool saw_tag_char = false; - - only_parenthesis = true; - - for (const char * c = p; *c != '\0'; c++) { - bool consumed = false; - - if (*c != '(' && *c != ')') - only_parenthesis = false; - - if (in_prefix) { - switch (*c) { - case '(': - break; - case '@': - expr << "(payee =~ /"; - found_specifier = true; - consumed = true; - break; - case '=': - expr << "(note =~ /"; - found_specifier = true; - consumed = true; - break; - case '%': - expr << "(note =~ /:"; - found_specifier = true; - saw_tag_char = true; - consumed = true; - break; - case '/': - case '_': - default: - if (! found_specifier) { - expr << "(account =~ /"; - found_specifier = true; - } - in_prefix = false; - break; - } - } else { - switch (*c) { - case ')': - if (! in_suffix) { - if (found_specifier) { - if (saw_tag_char) - expr << ':'; - expr << "/)"; - } - in_suffix = true; - } - break; - default: - if (in_suffix) - throw_(parse_error, "Invalid text in specification argument"); - break; - } - } - - if (! consumed) - expr << *c; - } - - if (! in_suffix) { - if (found_specifier) { - if (saw_tag_char) - expr << ':'; - expr << "/)"; - } - } - } - - begin++; - } - - DEBUG("report.predicate", "Regexp predicate expression = " << expr.str()); - - return expr.str(); - } - template <class Type = xact_t, class handler_ptr = xact_handler_ptr, void (report_t::*report_method)(handler_ptr) = @@ -273,8 +153,8 @@ namespace { if (args.value().size() > 0) report.append_predicate - (args_to_predicate(args.value().as_sequence().begin(), - args.value().as_sequence().end())); + (args_to_predicate_expr(args.value().as_sequence().begin(), + args.value().as_sequence().end())); (report.*report_method)(handler_ptr(handler)); |