blob: 7110eb7c70a731b0a2882ba7f5a2b476acbadb9a [file] [log] [blame]
/*
* Copyright (C) 2009 Google Inc. All rights reserved.
* Copyright (C) 2014 University of Washington. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "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 COPYRIGHT
* OWNER OR CONTRIBUTORS 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.
*/
#ifndef InspectorValues_h
#define InspectorValues_h
#include "JSExportMacros.h"
#include <wtf/Assertions.h>
#include <wtf/HashMap.h>
#include <wtf/RefCounted.h>
#include <wtf/Vector.h>
#include <wtf/text/StringHash.h>
#include <wtf/text/WTFString.h>
namespace Inspector {
class InspectorArray;
class InspectorArrayBase;
class InspectorObject;
class InspectorObjectBase;
class JS_EXPORT_PRIVATE InspectorValue : public RefCounted<InspectorValue> {
public:
static const int maxDepth = 1000;
InspectorValue()
: m_type(Type::Null) { }
virtual ~InspectorValue() { }
static Ref<InspectorValue> null();
enum class Type {
Null = 0,
Boolean,
Double,
Integer,
String,
Object,
Array
};
Type type() const { return m_type; }
bool isNull() const { return m_type == Type::Null; }
virtual bool asBoolean(bool&) const;
virtual bool asInteger(int&) const;
virtual bool asInteger(unsigned&) const;
virtual bool asInteger(long&) const;
virtual bool asInteger(long long&) const;
virtual bool asInteger(unsigned long&) const;
virtual bool asInteger(unsigned long long&) const;
virtual bool asDouble(double&) const;
virtual bool asDouble(float&) const;
virtual bool asString(String&) const;
virtual bool asValue(RefPtr<InspectorValue>&);
virtual bool asObject(RefPtr<InspectorObject>&);
virtual bool asArray(RefPtr<InspectorArray>&);
static bool parseJSON(const String& jsonInput, RefPtr<InspectorValue>& output);
String toJSONString() const;
virtual void writeJSON(StringBuilder& output) const;
protected:
explicit InspectorValue(Type type) : m_type(type) { }
private:
Type m_type;
};
class JS_EXPORT_PRIVATE InspectorBasicValue : public InspectorValue {
public:
static Ref<InspectorBasicValue> create(bool);
static Ref<InspectorBasicValue> create(int);
static Ref<InspectorBasicValue> create(double);
virtual bool asBoolean(bool&) const override;
// Numbers from the frontend are always parsed as doubles, so we allow
// clients to convert to integral values with this function.
virtual bool asInteger(int&) const override;
virtual bool asInteger(unsigned&) const override;
virtual bool asInteger(long&) const override;
virtual bool asInteger(long long&) const override;
virtual bool asInteger(unsigned long&) const override;
virtual bool asInteger(unsigned long long&) const override;
virtual bool asDouble(double&) const override;
virtual bool asDouble(float&) const override;
virtual void writeJSON(StringBuilder& output) const override;
private:
explicit InspectorBasicValue(bool value)
: InspectorValue(Type::Boolean)
, m_booleanValue(value) { }
explicit InspectorBasicValue(int value)
: InspectorValue(Type::Integer)
, m_doubleValue(static_cast<double>(value)) { }
explicit InspectorBasicValue(double value)
: InspectorValue(Type::Double)
, m_doubleValue(value) { }
union {
bool m_booleanValue;
double m_doubleValue;
};
};
class JS_EXPORT_PRIVATE InspectorString : public InspectorValue {
public:
static Ref<InspectorString> create(const String&);
static Ref<InspectorString> create(const char*);
virtual bool asString(String& output) const override;
virtual void writeJSON(StringBuilder& output) const override;
private:
explicit InspectorString(const String& value)
: InspectorValue(Type::String)
, m_stringValue(value) { }
explicit InspectorString(const char* value)
: InspectorValue(Type::String)
, m_stringValue(value) { }
String m_stringValue;
};
class JS_EXPORT_PRIVATE InspectorObjectBase : public InspectorValue {
private:
typedef HashMap<String, RefPtr<InspectorValue>> Dictionary;
public:
typedef Dictionary::iterator iterator;
typedef Dictionary::const_iterator const_iterator;
InspectorObject* openAccessors();
protected:
virtual ~InspectorObjectBase();
virtual bool asObject(RefPtr<InspectorObject>& output) override;
// FIXME: use templates to reduce the amount of duplicated set*() methods.
void setBoolean(const String& name, bool);
void setInteger(const String& name, int);
void setDouble(const String& name, double);
void setString(const String& name, const String&);
void setValue(const String& name, RefPtr<InspectorValue>&&);
void setObject(const String& name, RefPtr<InspectorObjectBase>&&);
void setArray(const String& name, RefPtr<InspectorArrayBase>&&);
iterator find(const String& name);
const_iterator find(const String& name) const;
// FIXME: use templates to reduce the amount of duplicated get*() methods.
bool getBoolean(const String& name, bool& output) const;
template<class T> bool getDouble(const String& name, T& output) const
{
RefPtr<InspectorValue> value;
if (!getValue(name, value))
return false;
return value->asDouble(output);
}
template<class T> bool getInteger(const String& name, T& output) const
{
RefPtr<InspectorValue> value;
if (!getValue(name, value))
return false;
return value->asInteger(output);
}
bool getString(const String& name, String& output) const;
bool getObject(const String& name, RefPtr<InspectorObject>&) const;
bool getArray(const String& name, RefPtr<InspectorArray>&) const;
bool getValue(const String& name, RefPtr<InspectorValue>&) const;
void remove(const String& name);
virtual void writeJSON(StringBuilder& output) const override;
iterator begin() { return m_data.begin(); }
iterator end() { return m_data.end(); }
const_iterator begin() const { return m_data.begin(); }
const_iterator end() const { return m_data.end(); }
int size() const { return m_data.size(); }
protected:
InspectorObjectBase();
private:
Dictionary m_data;
Vector<String> m_order;
};
class InspectorObject : public InspectorObjectBase {
public:
static JS_EXPORT_PRIVATE Ref<InspectorObject> create();
using InspectorObjectBase::asObject;
using InspectorObjectBase::setBoolean;
using InspectorObjectBase::setInteger;
using InspectorObjectBase::setDouble;
using InspectorObjectBase::setString;
using InspectorObjectBase::setValue;
using InspectorObjectBase::setObject;
using InspectorObjectBase::setArray;
using InspectorObjectBase::find;
using InspectorObjectBase::getBoolean;
using InspectorObjectBase::getInteger;
using InspectorObjectBase::getDouble;
using InspectorObjectBase::getString;
using InspectorObjectBase::getObject;
using InspectorObjectBase::getArray;
using InspectorObjectBase::getValue;
using InspectorObjectBase::remove;
using InspectorObjectBase::begin;
using InspectorObjectBase::end;
using InspectorObjectBase::size;
};
class JS_EXPORT_PRIVATE InspectorArrayBase : public InspectorValue {
public:
typedef Vector<RefPtr<InspectorValue>>::iterator iterator;
typedef Vector<RefPtr<InspectorValue>>::const_iterator const_iterator;
unsigned length() const { return static_cast<unsigned>(m_data.size()); }
protected:
virtual ~InspectorArrayBase();
virtual bool asArray(RefPtr<InspectorArray>&) override;
void pushBoolean(bool);
void pushInteger(int);
void pushDouble(double);
void pushString(const String&);
void pushValue(RefPtr<InspectorValue>&&);
void pushObject(RefPtr<InspectorObjectBase>&&);
void pushArray(RefPtr<InspectorArrayBase>&&);
RefPtr<InspectorValue> get(size_t index) const;
virtual void writeJSON(StringBuilder& output) const override;
iterator begin() { return m_data.begin(); }
iterator end() { return m_data.end(); }
const_iterator begin() const { return m_data.begin(); }
const_iterator end() const { return m_data.end(); }
protected:
InspectorArrayBase();
private:
Vector<RefPtr<InspectorValue>> m_data;
};
class InspectorArray : public InspectorArrayBase {
public:
static JS_EXPORT_PRIVATE Ref<InspectorArray> create();
using InspectorArrayBase::asArray;
using InspectorArrayBase::pushBoolean;
using InspectorArrayBase::pushInteger;
using InspectorArrayBase::pushDouble;
using InspectorArrayBase::pushString;
using InspectorArrayBase::pushValue;
using InspectorArrayBase::pushObject;
using InspectorArrayBase::pushArray;
using InspectorArrayBase::get;
using InspectorArrayBase::begin;
using InspectorArrayBase::end;
};
inline InspectorObjectBase::iterator InspectorObjectBase::find(const String& name)
{
return m_data.find(name);
}
inline InspectorObjectBase::const_iterator InspectorObjectBase::find(const String& name) const
{
return m_data.find(name);
}
inline void InspectorObjectBase::setBoolean(const String& name, bool value)
{
setValue(name, InspectorBasicValue::create(value));
}
inline void InspectorObjectBase::setInteger(const String& name, int value)
{
setValue(name, InspectorBasicValue::create(value));
}
inline void InspectorObjectBase::setDouble(const String& name, double value)
{
setValue(name, InspectorBasicValue::create(value));
}
inline void InspectorObjectBase::setString(const String& name, const String& value)
{
setValue(name, InspectorString::create(value));
}
inline void InspectorObjectBase::setValue(const String& name, RefPtr<InspectorValue>&& value)
{
ASSERT(value);
if (m_data.set(name, WTF::move(value)).isNewEntry)
m_order.append(name);
}
inline void InspectorObjectBase::setObject(const String& name, RefPtr<InspectorObjectBase>&& value)
{
ASSERT(value);
if (m_data.set(name, WTF::move(value)).isNewEntry)
m_order.append(name);
}
inline void InspectorObjectBase::setArray(const String& name, RefPtr<InspectorArrayBase>&& value)
{
ASSERT(value);
if (m_data.set(name, WTF::move(value)).isNewEntry)
m_order.append(name);
}
inline void InspectorArrayBase::pushBoolean(bool value)
{
m_data.append(InspectorBasicValue::create(value));
}
inline void InspectorArrayBase::pushInteger(int value)
{
m_data.append(InspectorBasicValue::create(value));
}
inline void InspectorArrayBase::pushDouble(double value)
{
m_data.append(InspectorBasicValue::create(value));
}
inline void InspectorArrayBase::pushString(const String& value)
{
m_data.append(InspectorString::create(value));
}
inline void InspectorArrayBase::pushValue(RefPtr<InspectorValue>&& value)
{
ASSERT(value);
m_data.append(WTF::move(value));
}
inline void InspectorArrayBase::pushObject(RefPtr<InspectorObjectBase>&& value)
{
ASSERT(value);
m_data.append(WTF::move(value));
}
inline void InspectorArrayBase::pushArray(RefPtr<InspectorArrayBase>&& value)
{
ASSERT(value);
m_data.append(WTF::move(value));
}
} // namespace Inspector
#endif // !defined(InspectorValues_h)