// String in object
if (is_val<value_string>(left_val) && is_val<value_object>(right_val)) {
auto key = left_val->as_string().str();
- auto & obj = right_val->as_object();
- bool has_key = obj.find(key) != obj.end();
+ bool has_key = right_val->has_key(key);
if (op.value == "in") {
return mk_val<value_bool>(has_key);
} else if (op.value == "not in") {
std::vector<value> items;
if (is_val<value_object>(iterable_val)) {
JJ_DEBUG("%s", "For loop over object keys");
- auto & obj = iterable_val->as_object();
+ auto & obj = iterable_val->as_ordered_object();
for (auto & p : obj) {
auto tuple = mk_val<value_array>();
if (iterable_val->val_obj.is_key_numeric) {
throw std::runtime_error("Cannot access object with non-string: got " + property->type());
}
auto key = property->as_string().str();
- auto & obj = object->as_object();
- auto it = obj.find(key);
- if (it != obj.end()) {
- val = it->second;
- } else {
+ val = object->at(key, val);
+ if (is_val<value_undefined>(val)) {
val = try_builtin_func(ctx, key, object, true);
}
JJ_DEBUG("Accessed property '%s' value, got type: %s", key.c_str(), val->type().c_str());
context(const context & parent) : context() {
// inherit variables (for example, when entering a new scope)
- auto & pvar = parent.env->as_object();
+ auto & pvar = parent.env->as_ordered_object();
for (const auto & pair : pvar) {
set_val(pair.first, pair.second);
}
if (args.count() == 3) {
default_val = args.get_pos(2);
}
- const auto & obj = args.get_pos(0)->as_object();
+ const value obj = args.get_pos(0);
std::string key = args.get_pos(1)->as_string().str();
- auto it = obj.find(key);
- if (it != obj.end()) {
- return it->second;
- } else {
- return default_val;
- }
+ return obj->at(key, default_val);
}},
{"keys", [](const func_args & args) -> value {
args.ensure_vals<value_object>();
- const auto & obj = args.get_pos(0)->as_object();
+ const auto & obj = args.get_pos(0)->as_ordered_object();
auto result = mk_val<value_array>();
for (const auto & pair : obj) {
result->push_back(mk_val<value_string>(pair.first));
}},
{"values", [](const func_args & args) -> value {
args.ensure_vals<value_object>();
- const auto & obj = args.get_pos(0)->as_object();
+ const auto & obj = args.get_pos(0)->as_ordered_object();
auto result = mk_val<value_array>();
for (const auto & pair : obj) {
result->push_back(pair.second);
}},
{"items", [](const func_args & args) -> value {
args.ensure_vals<value_object>();
- const auto & obj = args.get_pos(0)->as_object();
+ const auto & obj = args.get_pos(0)->as_ordered_object();
auto result = mk_val<value_array>();
for (const auto & pair : obj) {
auto item = mk_val<value_array>();
{"string", tojson},
{"length", [](const func_args & args) -> value {
args.ensure_vals<value_object>();
- const auto & obj = args.get_pos(0)->as_object();
+ const auto & obj = args.get_pos(0)->as_ordered_object();
return mk_val<value_int>(static_cast<int64_t>(obj.size()));
}},
{"tojson", [](const func_args & args) -> value {
// FIXME: sorting is currently always case sensitive
//const bool case_sensitive = val_case->as_bool(); // undefined == false
const bool reverse = val_reverse->as_bool(); // undefined == false
- if (!val_by->is_undefined()) {
- throw not_implemented_exception("dictsort by key not implemented");
- }
- if (reverse) {
- throw not_implemented_exception("dictsort reverse not implemented");
- }
- value_t::map obj = val_input->val_obj; // copy
- std::sort(obj.ordered.begin(), obj.ordered.end(), [&](const auto & a, const auto & b) {
- return a.first < b.first;
+ const bool by_value = is_val<value_string>(val_by) && val_by->as_string().str() == "value" ? true : false;
+ auto result = mk_val<value_object>(val_input); // copy
+ std::sort(result->val_obj.ordered.begin(), result->val_obj.ordered.end(), [&](const auto & a, const auto & b) {
+ if (by_value) {
+ return value_compare(a.second, b.second, reverse ? value_compare_op::gt : value_compare_op::lt);
+ } else {
+ return reverse ? a.first > b.first : a.first < b.first;
+ }
});
- auto result = mk_val<value_object>();
- result->val_obj = std::move(obj);
return result;
}},
{"join", [](const func_args &) -> value {
}
oss << "]";
} else if (is_val<value_object>(val)) {
- const auto & obj = val->val_obj.ordered; // IMPORTANT: need to keep exact order
+ const auto & obj = val->as_ordered_object(); // IMPORTANT: need to keep exact order
oss << "{";
if (!obj.empty()) {
oss << newline();
virtual string as_string() const { throw std::runtime_error(type() + " is not a string value"); }
virtual bool as_bool() const { throw std::runtime_error(type() + " is not a bool value"); }
virtual const std::vector<value> & as_array() const { throw std::runtime_error(type() + " is not an array value"); }
- virtual const std::map<std::string, value> & as_object() const { throw std::runtime_error(type() + " is not an object value"); }
+ virtual const std::vector<std::pair<std::string, value>> & as_ordered_object() const { throw std::runtime_error(type() + " is not an object value"); }
virtual value invoke(const func_args &) const { throw std::runtime_error(type() + " is not a function value"); }
virtual bool is_none() const { return false; }
virtual bool is_undefined() const { return false; }
throw std::runtime_error("No builtins available for type " + type());
}
+ virtual bool has_key(const std::string & key) {
+ return val_obj.unordered.find(key) != val_obj.unordered.end();
+ }
virtual value & at(const std::string & key, value & default_val) {
auto it = val_obj.unordered.find(key);
if (it == val_obj.unordered.end()) {
val_obj.insert(pair.first, pair.second);
}
}
+ value_object_t(const std::vector<std::pair<std::string, value>> & obj) {
+ for (const auto & pair : obj) {
+ val_obj.insert(pair.first, pair.second);
+ }
+ }
void insert(const std::string & key, const value & val) {
val_obj.insert(key, val);
}
virtual std::string type() const override { return "Object"; }
- virtual const std::map<std::string, value> & as_object() const override { return val_obj.unordered; }
+ virtual const std::vector<std::pair<std::string, value>> & as_ordered_object() const override { return val_obj.ordered; }
virtual bool as_bool() const override {
return !val_obj.unordered.empty();
}