mysqlshdk/scripting/polyglot/native_wrappers/polyglot_map_wrapper.cc (119 lines of code) (raw):
/*
* Copyright (c) 2014, 2024, Oracle and/or its affiliates.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License, version 2.0,
* as published by the Free Software Foundation.
*
* This program is designed to work with certain software (including
* but not limited to OpenSSL) that is licensed under separate terms,
* as designated in a particular file or component or in included license
* documentation. The authors of MySQL hereby grant you an additional
* permission to link the program and your derivative works with the
* separately licensed software that they have either included with
* the program or referenced in the documentation.
*
* This program 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, version 2.0, for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "mysqlshdk/scripting/polyglot/native_wrappers/polyglot_map_wrapper.h"
#include <cassert>
#include <exception>
#include <memory>
#include <stdexcept>
#include <vector>
#include "mysqlshdk/include/scripting/types.h"
#include "mysqlshdk/scripting/polyglot/languages/polyglot_language.h"
#include "mysqlshdk/scripting/polyglot/native_wrappers/polyglot_collectable.h"
#include "mysqlshdk/scripting/polyglot/native_wrappers/polyglot_iterator_wrapper.h"
#include "mysqlshdk/scripting/polyglot/utils/polyglot_error.h"
#include "mysqlshdk/scripting/polyglot/utils/polyglot_utils.h"
namespace shcore {
namespace polyglot {
namespace {
class Map_iterator : public IPolyglot_iterator {
public:
Map_iterator(Value::Map_type::iterator begin, Value::Map_type::iterator end)
: m_index(begin), m_end(end) {}
~Map_iterator() override = default;
bool has_next() const override { return m_index != m_end; }
shcore::Value get_next() override {
shcore::Array_t item;
if (m_index != m_end) {
item = shcore::make_array();
item->push_back(shcore::Value(m_index->first));
item->push_back(m_index->second);
m_index++;
}
return shcore::Value(std::move(item));
}
private:
Value::Map_type::iterator m_index;
Value::Map_type::iterator m_end;
};
struct Get_member_keys {
static constexpr const char *name = "getMemberKeys";
static poly_value callback(const std::shared_ptr<Polyglot_language> &language,
const Polyglot_map_wrapper::Native_ptr &map) {
std::vector<poly_value> keys;
for (const auto &m : (*map)) {
keys.push_back(
poly_string(language->thread(), language->context(), m.first));
}
return poly_array(language->thread(), language->context(), keys);
}
};
struct Has_member {
static constexpr const char *name = "hasMember";
static constexpr std::size_t argc = 1;
static Value callback(const Polyglot_map_wrapper::Native_ptr &map,
const Argument_list &argv) {
return Value(map->has_key(argv[0].as_string()));
}
};
struct Get_member {
static constexpr const char *name = "getMember";
static constexpr std::size_t argc = 1;
static Value callback(const Polyglot_map_wrapper::Native_ptr &map,
const Argument_list &argv) {
auto member = argv[0].as_string();
if (map->has_key(member)) {
return map->at(member);
} else {
return {};
}
}
};
struct Put_member {
static constexpr const char *name = "putMember";
static constexpr std::size_t argc = 2;
static Value callback(const Polyglot_map_wrapper::Native_ptr &map,
const Argument_list &argv) {
auto member = argv[0].as_string();
(*map)[member] = argv[1];
return {};
}
};
struct Remove {
static constexpr const char *name = "remove";
static constexpr std::size_t argc = 1;
static Value callback(const Polyglot_map_wrapper::Native_ptr &map,
const Argument_list &argv) {
bool was_removed = false;
if (const auto it = map->find(argv[0].as_string()); it != map->end()) {
map->erase(it);
was_removed = true;
}
return Value(was_removed);
}
};
struct Get_iterator {
static constexpr const char *name = "getIterator";
static poly_value callback(const std::shared_ptr<Polyglot_language> &language,
const Polyglot_map_wrapper::Native_ptr &map) {
return Polyglot_iterator_wrapper(language).wrap(
std::make_shared<Map_iterator>(map->begin(), map->end()));
}
};
} // namespace
Polyglot_map_wrapper::Polyglot_map_wrapper(
std::weak_ptr<Polyglot_language> language)
: Polyglot_native_wrapper(std::move(language)) {}
poly_value Polyglot_map_wrapper::create_wrapper(
poly_thread thread, poly_context context, ICollectable *collectable) const {
poly_value poly_map;
throw_if_error(
poly_create_proxy_iterable_object, thread, context, collectable,
&Polyglot_map_wrapper::polyglot_handler_no_args<Get_member_keys>,
&Polyglot_map_wrapper::native_handler_fixed_args<Has_member>,
&Polyglot_map_wrapper::native_handler_fixed_args<Put_member>,
&Polyglot_map_wrapper::native_handler_fixed_args<Get_member>,
&Polyglot_map_wrapper::native_handler_fixed_args<Remove>,
&Polyglot_map_wrapper::polyglot_handler_no_args<Get_iterator>,
&Polyglot_map_wrapper::handler_release_collectable, &poly_map);
return poly_map;
}
} // namespace polyglot
} // namespace shcore