/* * Copyright (C) 2016 Charybdis Development Team * Copyright (C) 2016 Jason Volk * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice is present in all copies. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #pragma once #define HAVE_IRCD_JS_VALUE_H namespace ircd { namespace js { // Fundamental string utils. This is placed here instead of string.h so // `struct value` can make direct conversions without depending on struct string type size_t native_size(const JSString *const &); size_t native(const JSString *const &, char *const &buf, const size_t &max); std::string native(const JSString *const &); struct value :JS::Rooted { using handle = JS::HandleValue; using handle_mutable = JS::MutableHandleValue; explicit operator std::string() const; explicit operator double() const; explicit operator uint64_t() const; explicit operator int64_t() const; explicit operator uint32_t() const; explicit operator int32_t() const; explicit operator uint16_t() const; explicit operator bool() const; explicit value(const std::string &); value(const char *const &); value(const nullptr_t &); value(const double &); value(const float &); value(const int32_t &); value(const bool &); value(const jsid &); value(JSObject &); value(JSObject *const &); value(JSString *const &); value(JSFunction *const &); value(JS::Symbol *const &); value(const JS::Value &); template value(const JS::Handle &h); template value(const JS::MutableHandle &h); template value(const JS::Rooted &r); template value(const JS::PersistentRooted &p); value(); value(value &&) noexcept; value(const value &) = delete; value &operator=(value &&) noexcept; friend JSType type(const value &); friend bool undefined(const value &); }; inline value::value() :JS::Rooted{*cx, JS::UndefinedValue()} { } inline value::value(value &&other) noexcept :JS::Rooted{*cx, other.get()} { other.set(JS::UndefinedValue()); } inline value & value::operator=(value &&other) noexcept { set(other.get()); other.set(JS::UndefinedValue()); return *this; } template value::value(const JS::PersistentRooted &r) :value(JS::Handle(r)) { } template value::value(const JS::Rooted &r) :value(JS::Handle(r)) { } template value::value(const JS::MutableHandle &h) :value(h.get()) { } template value::value(const JS::Handle &h) :value(h.get()) { } inline value::value(const JS::Value &val) :JS::Rooted{*cx, val} { } inline value::value(JS::Symbol *const &val) :JS::Rooted{*cx, JS::SymbolValue(val)} { } inline value::value(JSObject *const &val) :JS::Rooted { *cx, val? JS::ObjectValue(*val) : throw internal_error("NULL JSObject") } { } inline value::value(JSObject &val) :JS::Rooted{*cx, JS::ObjectValue(val)} { } inline value::value(JSString *const &val) :JS::Rooted{*cx, JS::StringValue(val)} { } inline value::value(JSFunction *const &val) :JS::Rooted{*cx} { auto *const obj(JS_GetFunctionObject(val)); if(unlikely(!obj)) throw type_error("Function cannot convert to Object"); set(JS::ObjectValue(*obj)); } inline value::value(const jsid &val) :JS::Rooted{*cx} { if(!JS_IdToValue(*cx, val, &(*this))) throw type_error("Failed to construct value from Id"); } inline value::value(const bool &b) :JS::Rooted{*cx, JS::BooleanValue(b)} { } inline value::value(const int32_t &val) :JS::Rooted{*cx, JS::Int32Value(val)} { } inline value::value(const float &val) :JS::Rooted{*cx, JS::Float32Value(val)} { } inline value::value(const double &val) :JS::Rooted{*cx, JS::DoubleValue(val)} { } inline value::value(const nullptr_t &) :JS::Rooted{*cx, JS::NullValue()} { } inline value::value(const std::string &s) :JS::Rooted { *cx, JS::StringValue(JS_NewStringCopyN(*cx, s.data(), s.size())) } { } inline value::value(const char *const &s) :JS::Rooted { *cx, !s? JS::NullValue() : JS::StringValue(JS_NewStringCopyZ(*cx, s)) } { } inline value::operator bool() const { return JS::ToBoolean(*this); } inline value::operator uint16_t() const { uint16_t ret; if(!JS::ToUint16(*cx, *this, &ret)) throw type_error("Failed cast to uint16_t"); return ret; } inline value::operator int32_t() const { int32_t ret; if(!JS::ToInt32(*cx, *this, &ret)) throw type_error("Failed cast to int32_t"); return ret; } inline value::operator uint32_t() const { uint32_t ret; if(!JS::ToUint32(*cx, *this, &ret)) throw type_error("Failed cast to uint32_t"); return ret; } inline value::operator int64_t() const { int64_t ret; if(!JS::ToInt64(*cx, *this, &ret)) throw type_error("Failed cast to int64_t"); return ret; } inline value::operator uint64_t() const { uint64_t ret; if(!JS::ToUint64(*cx, *this, &ret)) throw type_error("Failed cast to uint64_t"); return ret; } inline value::operator double() const { double ret; if(!JS::ToNumber(*cx, *this, &ret)) throw type_error("Failed cast to double"); return ret; } inline value::operator std::string() const { const auto s(JS::ToString(*cx, *this)); return s? native(s) : throw type_error("Failed to cast to string"); } inline JSType type(const value &val) { return JS_TypeOfValue(*cx, val); } inline bool undefined(const value &val) { return type(val) == JSTYPE_VOID; } } // namespace js } // namespace ircd