// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "content/renderer/v8_value_converter_impl.h" #include #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "base/values.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebArrayBuffer.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebArrayBufferView.h" #include "v8/include/v8.h" using base::BinaryValue; using base::DictionaryValue; using base::ListValue; using base::StringValue; using base::Value; namespace content { V8ValueConverter* V8ValueConverter::create() { return new V8ValueConverterImpl(); } } V8ValueConverterImpl::V8ValueConverterImpl() : undefined_allowed_(false), date_allowed_(false), regexp_allowed_(false), strip_null_from_objects_(false) { } bool V8ValueConverterImpl::GetUndefinedAllowed() const { return undefined_allowed_; } void V8ValueConverterImpl::SetUndefinedAllowed(bool val) { undefined_allowed_ = val; } bool V8ValueConverterImpl::GetDateAllowed() const { return date_allowed_; } void V8ValueConverterImpl::SetDateAllowed(bool val) { date_allowed_ = val; } bool V8ValueConverterImpl::GetRegexpAllowed() const { return regexp_allowed_; } void V8ValueConverterImpl::SetRegexpAllowed(bool val) { regexp_allowed_ = val; } bool V8ValueConverterImpl::GetStripNullFromObjects() const { return strip_null_from_objects_; } void V8ValueConverterImpl::SetStripNullFromObjects(bool val) { strip_null_from_objects_ = val; } v8::Handle V8ValueConverterImpl::ToV8Value( const Value* value, v8::Handle context) const { v8::Context::Scope context_scope(context); v8::HandleScope handle_scope; return handle_scope.Close(ToV8ValueImpl(value)); } Value* V8ValueConverterImpl::FromV8Value( v8::Handle val, v8::Handle context) const { v8::Context::Scope context_scope(context); v8::HandleScope handle_scope; return FromV8ValueImpl(val); } v8::Handle V8ValueConverterImpl::ToV8ValueImpl( const Value* value) const { CHECK(value); switch (value->GetType()) { case Value::TYPE_NULL: return v8::Null(); case Value::TYPE_BOOLEAN: { bool val = false; CHECK(value->GetAsBoolean(&val)); return v8::Boolean::New(val); } case Value::TYPE_INTEGER: { int val = 0; CHECK(value->GetAsInteger(&val)); return v8::Integer::New(val); } case Value::TYPE_DOUBLE: { double val = 0.0; CHECK(value->GetAsDouble(&val)); return v8::Number::New(val); } case Value::TYPE_STRING: { std::string val; CHECK(value->GetAsString(&val)); return v8::String::New(val.c_str(), val.length()); } case Value::TYPE_LIST: return ToV8Array(static_cast(value)); case Value::TYPE_DICTIONARY: return ToV8Object(static_cast(value)); case Value::TYPE_BINARY: return ToArrayBuffer(static_cast(value)); default: LOG(ERROR) << "Unexpected value type: " << value->GetType(); return v8::Null(); } } v8::Handle V8ValueConverterImpl::ToV8Array( const ListValue* val) const { v8::Handle result(v8::Array::New(val->GetSize())); for (size_t i = 0; i < val->GetSize(); ++i) { Value* child = NULL; CHECK(val->Get(i, &child)); v8::Handle child_v8 = ToV8ValueImpl(child); CHECK(!child_v8.IsEmpty()); v8::TryCatch try_catch; result->Set(static_cast(i), child_v8); if (try_catch.HasCaught()) LOG(ERROR) << "Setter for index " << i << " threw an exception."; } return result; } v8::Handle V8ValueConverterImpl::ToV8Object( const DictionaryValue* val) const { v8::Handle result(v8::Object::New()); for (DictionaryValue::key_iterator iter = val->begin_keys(); iter != val->end_keys(); ++iter) { Value* child = NULL; CHECK(val->GetWithoutPathExpansion(*iter, &child)); const std::string& key = *iter; v8::Handle child_v8 = ToV8ValueImpl(child); CHECK(!child_v8.IsEmpty()); v8::TryCatch try_catch; result->Set(v8::String::New(key.c_str(), key.length()), child_v8); if (try_catch.HasCaught()) { LOG(ERROR) << "Setter for property " << key.c_str() << " threw an " << "exception."; } } return result; } v8::Handle V8ValueConverterImpl::ToArrayBuffer( const BinaryValue* value) const { WebKit::WebArrayBuffer buffer = WebKit::WebArrayBuffer::create(value->GetSize(), 1); memcpy(buffer.data(), value->GetBuffer(), value->GetSize()); return buffer.toV8Value(); } Value* V8ValueConverterImpl::FromV8ValueImpl(v8::Handle val) const { CHECK(!val.IsEmpty()); if (val->IsNull()) return Value::CreateNullValue(); if (val->IsBoolean()) return Value::CreateBooleanValue(val->ToBoolean()->Value()); if (val->IsInt32()) return Value::CreateIntegerValue(val->ToInt32()->Value()); if (val->IsNumber()) return Value::CreateDoubleValue(val->ToNumber()->Value()); if (val->IsString()) { v8::String::Utf8Value utf8(val->ToString()); return Value::CreateStringValue(std::string(*utf8, utf8.length())); } if (undefined_allowed_ && val->IsUndefined()) return Value::CreateNullValue(); if (date_allowed_ && val->IsDate()) { v8::Date* date = v8::Date::Cast(*val); return Value::CreateDoubleValue(date->NumberValue() / 1000.0); } if (regexp_allowed_ && val->IsRegExp()) { return Value::CreateStringValue( *v8::String::Utf8Value(val->ToString())); } // v8::Value doesn't have a ToArray() method for some reason. if (val->IsArray()) return FromV8Array(val.As()); if (val->IsObject()) { BinaryValue* binary_value = FromV8Buffer(val); if (binary_value) { return binary_value; } else { return FromV8Object(val->ToObject()); } } LOG(ERROR) << "Unexpected v8 value type encountered."; return Value::CreateNullValue(); } ListValue* V8ValueConverterImpl::FromV8Array(v8::Handle val) const { ListValue* result = new ListValue(); for (uint32 i = 0; i < val->Length(); ++i) { v8::TryCatch try_catch; v8::Handle child_v8 = val->Get(i); if (try_catch.HasCaught()) { LOG(ERROR) << "Getter for index " << i << " threw an exception."; child_v8 = v8::Null(); } // TODO(aa): It would be nice to support getters, but we need // http://code.google.com/p/v8/issues/detail?id=1342 to do it properly. if (!val->HasRealIndexedProperty(i)) continue; Value* child = FromV8ValueImpl(child_v8); CHECK(child); result->Append(child); } return result; } base::BinaryValue* V8ValueConverterImpl::FromV8Buffer( v8::Handle val) const { char* data = NULL; size_t length = 0; WebKit::WebArrayBuffer* array_buffer = WebKit::WebArrayBuffer::createFromV8Value(val); if (array_buffer) { data = reinterpret_cast(array_buffer->data()); length = array_buffer->byteLength(); } else { WebKit::WebArrayBufferView* view = WebKit::WebArrayBufferView::createFromV8Value(val); if (view) { data = reinterpret_cast(view->baseAddress()) + view->byteOffset(); length = view->byteLength(); } } if (data) return base::BinaryValue::CreateWithCopiedBuffer(data, length); else return NULL; } DictionaryValue* V8ValueConverterImpl::FromV8Object( v8::Handle val) const { scoped_ptr result(new DictionaryValue()); v8::Handle property_names(val->GetPropertyNames()); for (uint32 i = 0; i < property_names->Length(); ++i) { v8::Handle name(property_names->Get(i).As()); // TODO(aa): It would be nice to support getters, but we need // http://code.google.com/p/v8/issues/detail?id=1342 to do it properly. if (!val->HasRealNamedProperty(name)) continue; v8::String::Utf8Value name_utf8(name->ToString()); v8::TryCatch try_catch; v8::Handle child_v8 = val->Get(name); if (try_catch.HasCaught()) { LOG(ERROR) << "Getter for property " << *name_utf8 << " threw an exception."; child_v8 = v8::Null(); } scoped_ptr child(FromV8ValueImpl(child_v8)); CHECK(child.get()); // Strip null if asked (and since undefined is turned into null, undefined // too). The use case for supporting this is JSON-schema support, // specifically for extensions, where "optional" JSON properties may be // represented as null, yet due to buggy legacy code elsewhere isn't // treated as such (potentially causing crashes). For example, the // "tabs.create" function takes an object as its first argument with an // optional "windowId" property. // // Given just // // tabs.create({}) // // this will work as expected on code that only checks for the existence of // a "windowId" property (such as that legacy code). However given // // tabs.create({windowId: null}) // // there *is* a "windowId" property, but since it should be an int, code // on the browser which doesn't additionally check for null will fail. // We can avoid all bugs related to this by stripping null. if (strip_null_from_objects_ && child->IsType(Value::TYPE_NULL)) continue; result->SetWithoutPathExpansion(std::string(*name_utf8, name_utf8.length()), child.release()); } return result.release(); }