mysqlshdk/scripting/polyglot/native_wrappers/polyglot_seekable_channel_wrapper.cc (127 lines of code) (raw):

/* * Copyright (c) 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_seekable_channel_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" namespace shcore { namespace polyglot { namespace { struct Is_open { static constexpr const char *name = "isOpen"; static poly_value callback( const std::shared_ptr<Polyglot_language> &language, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel) { return poly_bool(language->thread(), language->context(), seekable_channel->is_open()); } }; struct Close { static constexpr const char *name = "close"; static poly_value callback( const std::shared_ptr<Polyglot_language> & /*language*/, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel) { seekable_channel->close(); return nullptr; } }; struct Read { static constexpr const char *name = "read"; static constexpr std::size_t argc = 2; static poly_value callback( const std::shared_ptr<Polyglot_language> &language, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel, const std::vector<poly_value> &argv) { size_t amount = to_int(language->thread(), argv[1]); std::string buffer; buffer.resize(amount); auto read = seekable_channel->read(&buffer[0], amount); throw_if_error(poly_write_to_byte_buffer, language->thread(), language->context(), argv[0], static_cast<const void *>(buffer.data()), read); return poly_int(language->thread(), language->context(), read); } }; struct Write { static constexpr const char *name = "write"; static constexpr std::size_t argc = 1; static poly_value callback( const std::shared_ptr<Polyglot_language> &language, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel, const std::vector<poly_value> & /*argv*/) { const char *buffer = nullptr; size_t amount = 0; int64_t size = seekable_channel->write(buffer, amount); return poly_int(language->thread(), language->context(), size); } }; struct Position { static constexpr const char *name = "position"; static poly_value callback( const std::shared_ptr<Polyglot_language> &language, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel) { return poly_int(language->thread(), language->context(), seekable_channel->position()); } }; struct Set_position { static constexpr const char *name = "setPosition"; static constexpr std::size_t argc = 1; static poly_value callback( const std::shared_ptr<Polyglot_language> &language, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel, const std::vector<poly_value> &argv) { seekable_channel->set_position(to_int(language->thread(), argv[0])); return Polyglot_seekable_channel_wrapper(language).wrap(seekable_channel); } }; struct Size { static constexpr const char *name = "size"; static poly_value callback( const std::shared_ptr<Polyglot_language> &language, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel) { poly_value value; poly_create_double(language->thread(), language->context(), static_cast<double>(seekable_channel->size()), &value); return value; } }; struct Truncate { static constexpr const char *name = "truncate"; static constexpr std::size_t argc = 1; static poly_value callback( const std::shared_ptr<Polyglot_language> &language, const Polyglot_seekable_channel_wrapper::Native_ptr &seekable_channel, const std::vector<poly_value> &argv) { seekable_channel->truncate(to_int(language->thread(), argv[0])); return Polyglot_seekable_channel_wrapper(language).wrap(seekable_channel); } }; } // namespace Polyglot_seekable_channel_wrapper::Polyglot_seekable_channel_wrapper( std::weak_ptr<Polyglot_language> language) : Polyglot_native_wrapper(std::move(language)) {} poly_value Polyglot_seekable_channel_wrapper::create_wrapper( poly_thread thread, poly_context context, ICollectable *collectable) const { poly_value poly_object; throw_if_error( poly_create_proxy_seekable_byte_channel, thread, context, collectable, &Polyglot_seekable_channel_wrapper::polyglot_handler_no_args<Is_open>, &Polyglot_seekable_channel_wrapper::polyglot_handler_no_args<Close>, &Polyglot_seekable_channel_wrapper::polyglot_handler_fixed_args<Read>, &Polyglot_seekable_channel_wrapper::polyglot_handler_fixed_args<Write>, &Polyglot_seekable_channel_wrapper::polyglot_handler_no_args<Position>, &Polyglot_seekable_channel_wrapper::polyglot_handler_fixed_args< Set_position>, &Polyglot_seekable_channel_wrapper::polyglot_handler_no_args<Size>, &Polyglot_seekable_channel_wrapper::polyglot_handler_fixed_args<Truncate>, &Polyglot_seekable_channel_wrapper::handler_release_collectable, &poly_object); return poly_object; } } // namespace polyglot } // namespace shcore