bool evaluate(const string &query) const { if(query.empty()) return true; lstring rules = string{query}.replace(" ", "").split(","); for(auto &rule : rules) { enum class Comparator : unsigned { ID, EQ, NE, LT, LE, GT, GE }; auto comparator = Comparator::ID; if(rule.wildcard("*!=*")) comparator = Comparator::NE; else if(rule.wildcard("*<=*")) comparator = Comparator::LE; else if(rule.wildcard("*>=*")) comparator = Comparator::GE; else if(rule.wildcard ("*=*")) comparator = Comparator::EQ; else if(rule.wildcard ("*<*")) comparator = Comparator::LT; else if(rule.wildcard ("*>*")) comparator = Comparator::GT; if(comparator == Comparator::ID) { if(find(rule).size()) continue; return false; } lstring side; switch(comparator) { case Comparator::EQ: side = rule.split<1> ("="); break; case Comparator::NE: side = rule.split<1>("!="); break; case Comparator::LT: side = rule.split<1> ("<"); break; case Comparator::LE: side = rule.split<1>("<="); break; case Comparator::GT: side = rule.split<1> (">"); break; case Comparator::GE: side = rule.split<1>(">="); break; } string data = text(); if(side(0).empty() == false) { auto result = find(side(0)); if(result.size() == 0) return false; data = result(0).data; } switch(comparator) { case Comparator::EQ: if(data.wildcard(side(1)) == true) continue; break; case Comparator::NE: if(data.wildcard(side(1)) == false) continue; break; case Comparator::LT: if(numeral(data) < numeral(side(1))) continue; break; case Comparator::LE: if(numeral(data) <= numeral(side(1))) continue; break; case Comparator::GT: if(numeral(data) > numeral(side(1))) continue; break; case Comparator::GE: if(numeral(data) >= numeral(side(1))) continue; break; } return false; } return true; } vector find(const string &query) const { vector result; lstring path = query.split("/"); string name = path.take(0), rule; unsigned lo = 0u, hi = ~0u; if(name.wildcard("*[*]")) { lstring side = name.split<1>("["); name = side(0); side = side(1).rtrim<1>("]").split<1>("-"); lo = side(0).empty() ? 0u : numeral(side(0)); hi = side(1).empty() ? ~0u : numeral(side(1)); } if(name.wildcard("*(*)")) { lstring side = name.split<1>("("); name = side(0); rule = side(1).rtrim<1>(")"); } unsigned position = 0; for(auto &node : children) { if(node.name.wildcard(name) == false) continue; if(node.evaluate(rule) == false) continue; bool inrange = position >= lo && position <= hi; position++; if(inrange == false) continue; if(path.size() == 0) result.append(node); else { auto list = node.find(path.concatenate("/")); for(auto &item : list) result.append(item); } } return result; }