mysqlshdk/scripting/polyglot/native_wrappers/polyglot_object_wrapper.cc (169 lines of code) (raw):
/*
* Copyright (c) 2024, 2025, 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_object_wrapper.h"
#include <algorithm>
#include <cassert>
#include <exception>
#include <memory>
#include <stdexcept>
#include <string>
#include <utility>
#include <vector>
#include "mysqlshdk/scripting/polyglot/utils/polyglot_api_clean.h"
#include "mysqlshdk/include/scripting/types.h"
#include "mysqlshdk/include/shellcore/interrupt_handler.h"
#include "mysqlshdk/libs/db/session.h" // mysqlshdk::db::Error
#include "mysqlshdk/libs/utils/debug.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 Object_iterator : public IPolyglot_iterator {
public:
Object_iterator(std::vector<std::string> &&members)
: m_members(std::move(members)),
m_index{m_members.begin()},
m_end(m_members.end()) {}
~Object_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));
m_index++;
}
return shcore::Value(item);
}
private:
std::vector<std::string> m_members;
std::vector<std::string>::iterator m_index;
std::vector<std::string>::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_object_wrapper::Native_ptr &object) {
auto member_list = object->get_members();
auto last = std::unique(member_list.begin(), member_list.end());
member_list.erase(last, member_list.end());
std::vector<poly_value> members;
for (const auto &member : member_list) {
members.push_back(
poly_string(language->thread(), language->context(), member));
}
return poly_array(language->thread(), language->context(), members);
}
};
struct Has_member {
static constexpr const char *name = "hasMember";
static constexpr std::size_t argc = 1;
static Value callback(const Polyglot_object_wrapper::Native_ptr &object,
const Argument_list &argv) {
bool result = object->has_member(argv[0].as_string());
if (!result && object->is_indexed()) {
try {
result = argv[0].as_uint() < static_cast<uint64_t>(object->length());
} catch (...) {
// NO-OP: was not a number, so no indexed access
}
}
return Value(result);
}
};
struct Get_member {
static constexpr const char *name = "getMember";
static constexpr std::size_t argc = 1;
static poly_value callback(const std::shared_ptr<Polyglot_language> &language,
const Polyglot_object_wrapper::Native_ptr &object,
const std::vector<poly_value> &argv) {
shcore::Value identifier = language->convert(argv[0]);
std::string prop = identifier.as_string();
if (object->has_method(prop)) {
return Polyglot_method_wrapper(language).wrap(
std::make_shared<Object_method>(object, prop));
} else if (object->has_member(prop)) {
return language->convert(object->get_member(prop));
} else if (object->is_indexed()) {
try {
auto index = static_cast<size_t>(identifier.as_uint());
if (index < object->length()) {
return language->convert(object->get_member(index));
}
} catch (...) {
// NO-OP: was not a number, so no indexed access
}
}
return language->undefined();
}
};
struct Put_member {
static constexpr const char *name = "putMember";
static constexpr std::size_t argc = 2;
static Value callback(const Polyglot_object_wrapper::Native_ptr &object,
const Argument_list &argv) {
object->set_member(argv[0].as_string(), argv[1]);
return {};
}
};
struct Get_iterator {
static constexpr const char *name = "getIterator";
static poly_value callback(
const std::shared_ptr<Polyglot_language> &language,
const Polyglot_object_wrapper::Native_ptr &object) {
return Polyglot_iterator_wrapper(language).wrap(
std::make_shared<Object_iterator>(object->get_members()));
}
};
struct Method_call {
static Value callback(
const Polyglot_method_wrapper::Native_ptr &object_method,
const shcore::Argument_list &argv) {
return object_method->object()->call(object_method->method(), argv);
}
};
} // namespace
Polyglot_object_wrapper::Polyglot_object_wrapper(
std::weak_ptr<Polyglot_language> language, bool indexed)
: Polyglot_native_wrapper(std::move(language)), m_indexed(indexed) {}
poly_value Polyglot_object_wrapper::create_wrapper(
poly_thread thread, poly_context context, ICollectable *collectable) const {
poly_value poly_object;
if (m_indexed) {
throw_if_error(
poly_create_proxy_iterable_object, thread, context, collectable,
&Polyglot_object_wrapper::polyglot_handler_no_args<Get_member_keys>,
&Polyglot_object_wrapper::native_handler_fixed_args<Has_member>,
&Polyglot_object_wrapper::native_handler_fixed_args<Put_member>,
&Polyglot_object_wrapper::polyglot_handler_fixed_args<Get_member>,
nullptr, // no remove member handler,
&Polyglot_object_wrapper::polyglot_handler_no_args<Get_iterator>,
&Polyglot_object_wrapper::handler_release_collectable, &poly_object);
} else {
throw_if_error(
poly_create_proxy_object, thread, context, collectable,
&Polyglot_object_wrapper::polyglot_handler_no_args<Get_member_keys>,
&Polyglot_object_wrapper::native_handler_fixed_args<Has_member>,
&Polyglot_object_wrapper::native_handler_fixed_args<Put_member>,
&Polyglot_object_wrapper::polyglot_handler_fixed_args<Get_member>,
nullptr, // no remove member handler,
&Polyglot_object_wrapper::handler_release_collectable, &poly_object);
}
return poly_object;
}
Polyglot_method_wrapper::Polyglot_method_wrapper(
std::weak_ptr<Polyglot_language> language)
: Polyglot_native_wrapper(std::move(language)) {}
poly_value Polyglot_method_wrapper::create_wrapper(
poly_thread thread, poly_context context, ICollectable *collectable) const {
poly_value function;
throw_if_error(
poly_create_proxy_function, thread, context,
&Polyglot_method_wrapper::native_handler_variable_args<Method_call>,
&Polyglot_method_wrapper::handler_release_collectable, collectable,
&function);
return function;
}
} // namespace polyglot
} // namespace shcore