Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(388)

Unified Diff: Source/bindings/v8/Dictionary.h

Issue 85263002: Improve handling of dictionary conversions. (Closed) Base URL: https://chromium.googlesource.com/chromium/blink.git@master
Patch Set: Have conversion methods take a context argument; elaborate error msgs further. Created 7 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « Source/bindings/tests/results/V8TestExtendedEvent.cpp ('k') | Source/bindings/v8/Dictionary.cpp » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: Source/bindings/v8/Dictionary.h
diff --git a/Source/bindings/v8/Dictionary.h b/Source/bindings/v8/Dictionary.h
index 2d50a037606f3f40115eaa924e05a331849f3a67..c8dbae2e9ccb0c82ffe2272419be7b062f254743 100644
--- a/Source/bindings/v8/Dictionary.h
+++ b/Source/bindings/v8/Dictionary.h
@@ -26,7 +26,11 @@
#ifndef Dictionary_h
#define Dictionary_h
+#include "bindings/v8/ExceptionMessages.h"
+#include "bindings/v8/ExceptionState.h"
#include "bindings/v8/ScriptValue.h"
+#include "bindings/v8/V8Binding.h"
+#include "bindings/v8/V8BindingMacros.h"
#include "core/events/EventListener.h"
#include "core/dom/MessagePort.h"
#include <v8.h>
@@ -95,11 +99,81 @@ public:
bool get(const String&, RefPtr<VoidCallback>&) const;
bool get(const String&, v8::Local<v8::Value>&) const;
+ class ConversionContext {
+ public:
+ ConversionContext(const String& interfaceName, const String& methodName, ExceptionState& exceptionState)
+ : m_interfaceName(interfaceName)
sof 2013/11/27 23:46:43 I notice https://codereview.chromium.org/87963002,
+ , m_methodName(methodName)
+ , m_exceptionState(exceptionState)
+ , m_dirty(true)
+ {
+ resetPerPropertyContext();
+ }
+
+ const String& interfaceName() const { return m_interfaceName; }
+ const String& methodName() const { return m_methodName; }
+ bool forConstructor() const { return m_methodName.isEmpty(); }
+ ExceptionState& exceptionState() const { return m_exceptionState; }
+
+ bool isNullable() const { return m_isNullable; }
+ String typeName() const { return m_propertyTypeName; }
+ IntegerConversionConfiguration numberConversion() const { return m_numberConversion; }
+
+ ConversionContext& withAttributes(bool, IntegerConversionConfiguration, const String&);
+ ConversionContext& withAttributes(bool, IntegerConversionConfiguration);
+ ConversionContext& withAttributes(bool, const String&);
+ ConversionContext& withAttributes(bool);
+ ConversionContext& withAttributes(const String&);
+
+ void throwTypeError(const String& detail);
+
+ void resetPerPropertyContext();
+
+ private:
+ const String m_interfaceName;
+ const String m_methodName;
+ ExceptionState& m_exceptionState;
+ bool m_dirty;
+
+ bool m_isNullable;
+ String m_propertyTypeName;
+ IntegerConversionConfiguration m_numberConversion;
+ };
+
+ class ConversionContextScope {
+ public:
+ ConversionContextScope(ConversionContext& context)
+ : m_context(context) { }
+ ~ConversionContextScope()
+ {
+ m_context.resetPerPropertyContext();
+ }
+ private:
+ ConversionContext& m_context;
+ };
+
+ bool convert(ConversionContext&, const String&, bool&) const;
+ bool convert(ConversionContext&, const String&, double&) const;
+ bool convert(ConversionContext&, const String&, String&) const;
+ bool convert(ConversionContext&, const String&, ScriptValue&) const;
+
+ template<typename IntegralType>
+ bool convert(ConversionContext &, const String&, IntegralType&) const;
+ bool convert(ConversionContext &, const String&, MessagePortArray&) const;
+ bool convert(ConversionContext &, const String&, HashSet<AtomicString>&) const;
+ bool convert(ConversionContext &, const String&, Dictionary&) const;
+ bool convert(ConversionContext &, const String&, Vector<String>&) const;
+ bool convert(ConversionContext &, const String&, ArrayValue&) const;
+ template<typename T>
+ bool convert(ConversionContext &, const String&, RefPtr<T>&) const;
+
bool getOwnPropertiesAsStringHashMap(HashMap<String, String>&) const;
bool getOwnPropertyNames(Vector<String>&) const;
bool getWithUndefinedOrNullCheck(const String&, String&) const;
+ bool hasProperty(const String&) const;
+
// Only allow inline allocation.
void* operator new(size_t, NotNullTag, void* location) { return location; }
@@ -113,6 +187,149 @@ private:
v8::Isolate* m_isolate;
};
+template<>
+struct NativeValueTraits<Dictionary> {
+ static inline Dictionary nativeValue(const v8::Handle<v8::Value>& value, v8::Isolate* isolate)
+ {
+ return Dictionary(value, isolate);
+ }
+};
+
+template <typename T>
+struct IntegralTypeTraits {
+};
+
+template <>
+struct IntegralTypeTraits<uint8_t> {
+ static inline uint8_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toUInt8(value, configuration, ok);
+ }
+ static const String typeName() { return "UInt8"; }
+};
+
+template <>
+struct IntegralTypeTraits<int8_t> {
+ static inline int8_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toInt8(value, configuration, ok);
+ }
+ static const String typeName() { return "Int8"; }
+};
+
+template <>
+struct IntegralTypeTraits<unsigned short> {
+ static inline uint16_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toUInt16(value, configuration, ok);
+ }
+ static const String typeName() { return "UInt16"; }
+};
+
+template <>
+struct IntegralTypeTraits<short> {
+ static inline int16_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toInt16(value, configuration, ok);
+ }
+ static const String typeName() { return "Int16"; }
+};
+
+template <>
+struct IntegralTypeTraits<unsigned> {
+ static inline uint32_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toUInt32(value, configuration, ok);
+ }
+ static const String typeName() { return "UInt32"; }
+};
+
+template <>
+struct IntegralTypeTraits<unsigned long> {
+ static inline uint32_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toUInt32(value, configuration, ok);
+ }
+ static const String typeName() { return "UInt32"; }
+};
+
+template <>
+struct IntegralTypeTraits<int> {
+ static inline int32_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toInt32(value, configuration, ok);
+ }
+ static const String typeName() { return "Int32"; }
+};
+
+template <>
+struct IntegralTypeTraits<long> {
+ static inline int32_t toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toInt32(value, configuration, ok);
+ }
+ static const String typeName() { return "Int32"; }
+};
+
+template <>
+struct IntegralTypeTraits<unsigned long long> {
+ static inline unsigned long long toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toUInt64(value, configuration, ok);
+ }
+ static const String typeName() { return "UInt64"; }
+};
+
+template <>
+struct IntegralTypeTraits<long long> {
+ static inline long long toIntegral(v8::Handle<v8::Value> value, IntegerConversionConfiguration configuration, bool& ok)
+ {
+ return toInt64(value, configuration, ok);
+ }
+ static const String typeName() { return "Int64"; }
+};
+
+template<typename T> bool Dictionary::convert(ConversionContext& context, const String& key, T& value) const
+{
+ ConversionContextScope scope(context);
+
+ v8::Local<v8::Value> v8Value;
+ if (!getKey(key, v8Value))
+ return true;
+
+ bool ok = false;
+ value = IntegralTypeTraits<T>::toIntegral(v8Value, context.numberConversion(), ok);
+ if (!ok) {
+ V8TRYCATCH_RETURN(v8::Local<v8::Number>, v8Number, v8Value->ToNumber(), false);
+ if (v8Number.IsEmpty()) {
+ context.throwTypeError(ExceptionMessages::incorrectPropertyType(key, "does not have type " + IntegralTypeTraits<T>::typeName() + "."));
+ } else {
+ ASSERT(context.numberConversion() == EnforceRange);
+ context.throwTypeError(ExceptionMessages::incorrectPropertyType(key, "is not a finite number."));
+ }
+ return false;
+ }
+ return true;
+}
+
+template<typename T> bool Dictionary::convert(ConversionContext& context, const String& key, RefPtr<T>& value) const
+{
+ ConversionContextScope scope(context);
+
+ if (!get(key, value))
+ return true;
+
+ if (!value) {
+ v8::Local<v8::Value> v8Value;
+ getKey(key, v8Value);
+ if (!(context.isNullable() && WebCore::isUndefinedOrNull(v8Value))) {
+ context.throwTypeError(ExceptionMessages::incorrectPropertyType(key, "does not have " + (context.typeName().isEmpty() ? String("the expected type.") : ("a " + context.typeName() + " type."))));
+ return false;
+ }
+ }
+ return true;
+}
+
}
#endif // Dictionary_h
« no previous file with comments | « Source/bindings/tests/results/V8TestExtendedEvent.cpp ('k') | Source/bindings/v8/Dictionary.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698