// 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 "webkit/plugins/ppapi/v8_var_converter.h" #include #include "base/logging.h" #include "base/memory/ref_counted.h" #include "base/memory/scoped_ptr.h" #include "base/values.h" #include "ppapi/c/pp_bool.h" #include "ppapi/c/pp_var.h" #include "ppapi/shared_impl/array_var.h" #include "ppapi/shared_impl/dictionary_var.h" #include "ppapi/shared_impl/ppapi_globals.h" #include "ppapi/shared_impl/proxy_lock.h" #include "ppapi/shared_impl/scoped_pp_var.h" #include "ppapi/shared_impl/test_globals.h" #include "ppapi/shared_impl/unittest_utils.h" #include "ppapi/shared_impl/var.h" #include "ppapi/shared_impl/var_tracker.h" #include "testing/gtest/include/gtest/gtest.h" #include "v8/include/v8.h" using ppapi::ArrayBufferVar; using ppapi::ArrayVar; using ppapi::DictionaryVar; using ppapi::PpapiGlobals; using ppapi::ProxyLock; using ppapi::ScopedPPVar; using ppapi::StringVar; using ppapi::TestGlobals; using ppapi::TestEqual; using ppapi::VarTracker; namespace webkit { namespace ppapi { namespace { // Maps PP_Var IDs to the V8 value handle they correspond to. typedef base::hash_map > VarHandleMap; bool Equals(const PP_Var& var, v8::Handle val, VarHandleMap* visited_ids) { if (::ppapi::VarTracker::IsVarTypeRefcounted(var.type)) { VarHandleMap::iterator it = visited_ids->find(var.value.as_id); if (it != visited_ids->end()) return it->second == val; (*visited_ids)[var.value.as_id] = val; } if (val->IsUndefined()) { return var.type == PP_VARTYPE_UNDEFINED; } else if (val->IsNull()) { return var.type == PP_VARTYPE_NULL; } else if (val->IsBoolean() || val->IsBooleanObject()) { return var.type == PP_VARTYPE_BOOL && PP_FromBool(val->ToBoolean()->Value()) == var.value.as_bool; } else if (val->IsInt32()) { return var.type == PP_VARTYPE_INT32 && val->ToInt32()->Value() == var.value.as_int; } else if (val->IsNumber() || val->IsNumberObject()) { return var.type == PP_VARTYPE_DOUBLE && fabs(val->ToNumber()->Value() - var.value.as_double) <= 1.0e-4; } else if (val->IsString() || val->IsStringObject()) { if (var.type != PP_VARTYPE_STRING) return false; StringVar* string_var = StringVar::FromPPVar(var); DCHECK(string_var); v8::String::Utf8Value utf8(val->ToString()); return std::string(*utf8, utf8.length()) == string_var->value(); } else if (val->IsArray()) { if (var.type != PP_VARTYPE_ARRAY) return false; ArrayVar* array_var = ArrayVar::FromPPVar(var); DCHECK(array_var); v8::Handle v8_array = val.As(); if (v8_array->Length() != array_var->elements().size()) return false; for (uint32 i = 0; i < v8_array->Length(); ++i) { v8::Handle child_v8 = v8_array->Get(i); if (!Equals(array_var->elements()[i].get(), child_v8, visited_ids)) return false; } return true; } else if (val->IsObject()) { if (var.type == PP_VARTYPE_ARRAY_BUFFER) { // TODO(raymes): Implement this when we have tests for array buffers. NOTIMPLEMENTED(); return false; } else { v8::Handle v8_object = val->ToObject(); if (var.type != PP_VARTYPE_DICTIONARY) return false; DictionaryVar* dict_var = DictionaryVar::FromPPVar(var); DCHECK(dict_var); v8::Handle property_names(v8_object->GetOwnPropertyNames()); if (property_names->Length() != dict_var->key_value_map().size()) return false; for (uint32 i = 0; i < property_names->Length(); ++i) { v8::Handle key(property_names->Get(i)); if (!key->IsString() && !key->IsNumber()) return false; v8::Handle child_v8 = v8_object->Get(key); v8::String::Utf8Value name_utf8(key->ToString()); ScopedPPVar release_key(ScopedPPVar::PassRef(), StringVar::StringToPPVar( std::string(*name_utf8, name_utf8.length()))); if (!dict_var->HasKey(release_key.get())) return false; ScopedPPVar release_value(ScopedPPVar::PassRef(), dict_var->Get(release_key.get())); if (!Equals(release_value.get(), child_v8, visited_ids)) return false; } return true; } } return false; } bool Equals(const PP_Var& var, v8::Handle val) { VarHandleMap var_handle_map; return Equals(var, val, &var_handle_map); } class V8VarConverterTest : public testing::Test { public: V8VarConverterTest() {} ~V8VarConverterTest() {} // testing::Test implementation. virtual void SetUp() { ProxyLock::Acquire(); v8::HandleScope handle_scope; v8::Handle global = v8::ObjectTemplate::New(); v8::Isolate* isolate = v8::Isolate::GetCurrent(); // TODO(marja): Use v8::Persistent::Reset here. context_ = v8::Persistent( isolate, v8::Context::New(isolate, NULL, global)); } virtual void TearDown() { context_.Dispose(context_->GetIsolate()); ASSERT_TRUE(PpapiGlobals::Get()->GetVarTracker()->GetLiveVars().empty()); ProxyLock::Release(); } protected: bool RoundTrip(const PP_Var& var, PP_Var* result) { V8VarConverter converter; v8::Context::Scope context_scope(context_); v8::HandleScope handle_scope; v8::Handle v8_result; if (!converter.ToV8Value(var, context_, &v8_result)) return false; if (!Equals(var, v8_result)) return false; if (!converter.FromV8Value(v8_result, context_, result)) return false; return true; } // Assumes a ref for var. bool RoundTripAndCompare(const PP_Var& var) { ScopedPPVar expected(ScopedPPVar::PassRef(), var); PP_Var actual_var; if (!RoundTrip(expected.get(), &actual_var)) return false; ScopedPPVar actual(ScopedPPVar::PassRef(), actual_var); return TestEqual(expected.get(), actual.get()); } // Context for the JavaScript in the test. v8::Persistent context_; private: TestGlobals globals_; }; } // namespace TEST_F(V8VarConverterTest, SimpleRoundTripTest) { EXPECT_TRUE(RoundTripAndCompare(PP_MakeUndefined())); EXPECT_TRUE(RoundTripAndCompare(PP_MakeNull())); EXPECT_TRUE(RoundTripAndCompare(PP_MakeInt32(100))); EXPECT_TRUE(RoundTripAndCompare(PP_MakeBool(PP_TRUE))); EXPECT_TRUE(RoundTripAndCompare(PP_MakeDouble(53.75))); } TEST_F(V8VarConverterTest, StringRoundTripTest) { EXPECT_TRUE(RoundTripAndCompare(StringVar::StringToPPVar(""))); EXPECT_TRUE(RoundTripAndCompare(StringVar::StringToPPVar("hello world!"))); } TEST_F(V8VarConverterTest, ArrayBufferRoundTripTest) { // TODO(raymes): Testing this here requires spinning up some of WebKit. // Work out how to do this. } TEST_F(V8VarConverterTest, DictionaryArrayRoundTripTest) { // Empty array. scoped_refptr array(new ArrayVar); ScopedPPVar release_array(ScopedPPVar::PassRef(), array->GetPPVar()); EXPECT_TRUE(RoundTripAndCompare(array->GetPPVar())); size_t index = 0; // Array with primitives. array->Set(index++, PP_MakeUndefined()); array->Set(index++, PP_MakeNull()); array->Set(index++, PP_MakeInt32(100)); array->Set(index++, PP_MakeBool(PP_FALSE)); array->Set(index++, PP_MakeDouble(0.123)); EXPECT_TRUE(RoundTripAndCompare(array->GetPPVar())); // Array with 2 references to the same string. ScopedPPVar release_string( ScopedPPVar::PassRef(), StringVar::StringToPPVar("abc")); array->Set(index++, release_string.get()); array->Set(index++, release_string.get()); EXPECT_TRUE(RoundTripAndCompare(array->GetPPVar())); // Array with nested array that references the same string. scoped_refptr array2(new ArrayVar); ScopedPPVar release_array2(ScopedPPVar::PassRef(), array2->GetPPVar()); array2->Set(0, release_string.get()); array->Set(index++, release_array2.get()); EXPECT_TRUE(RoundTripAndCompare(array->GetPPVar())); // Empty dictionary. scoped_refptr dictionary(new DictionaryVar); ScopedPPVar release_dictionary(ScopedPPVar::PassRef(), dictionary->GetPPVar()); EXPECT_TRUE(RoundTripAndCompare(dictionary->GetPPVar())); // Dictionary with primitives. dictionary->SetWithStringKey("1", PP_MakeUndefined()); dictionary->SetWithStringKey("2", PP_MakeNull()); dictionary->SetWithStringKey("3", PP_MakeInt32(-100)); dictionary->SetWithStringKey("4", PP_MakeBool(PP_TRUE)); dictionary->SetWithStringKey("5", PP_MakeDouble(-103.52)); EXPECT_TRUE(RoundTripAndCompare(dictionary->GetPPVar())); // Dictionary with 2 references to the same string. dictionary->SetWithStringKey("6", release_string.get()); dictionary->SetWithStringKey("7", release_string.get()); EXPECT_TRUE(RoundTripAndCompare(dictionary->GetPPVar())); // Dictionary with nested dictionary that references the same string. scoped_refptr dictionary2(new DictionaryVar); ScopedPPVar release_dictionary2(ScopedPPVar::PassRef(), dictionary2->GetPPVar()); dictionary2->SetWithStringKey("abc", release_string.get()); dictionary->SetWithStringKey("8", release_dictionary2.get()); EXPECT_TRUE(RoundTripAndCompare(dictionary->GetPPVar())); // Array with dictionary. array->Set(index++, release_dictionary.get()); EXPECT_TRUE(RoundTripAndCompare(array->GetPPVar())); // Array with dictionary with array. array2->Set(0, PP_MakeInt32(100)); dictionary->SetWithStringKey("9", release_array2.get()); EXPECT_TRUE(RoundTripAndCompare(array->GetPPVar())); // Array <-> dictionary cycle. dictionary->SetWithStringKey("10", release_array.get()); PP_Var result_var; EXPECT_TRUE(RoundTrip(release_dictionary.get(), &result_var)); ScopedPPVar result = ScopedPPVar(ScopedPPVar::PassRef(), result_var); EXPECT_TRUE(TestEqual(release_dictionary.get(), result.get())); // Break the cycle. // TODO(raymes): We need some better machinery for releasing vars with // cycles. Remove the code below once we have that. dictionary->DeleteWithStringKey("10"); DictionaryVar* result_dictionary = DictionaryVar::FromPPVar(result.get()); result_dictionary->DeleteWithStringKey("10"); // Array with self references. array->Set(index, release_array.get()); EXPECT_TRUE(RoundTrip(release_array.get(), &result_var)); result = ScopedPPVar(ScopedPPVar::PassRef(), result_var); EXPECT_TRUE(TestEqual(release_array.get(), result.get())); // Break the self reference. array->Set(index, PP_MakeUndefined()); ArrayVar* result_array = ArrayVar::FromPPVar(result.get()); result_array->Set(index, PP_MakeUndefined()); } TEST_F(V8VarConverterTest, StrangeDictionaryKeyTest) { { // Test keys with '.'. scoped_refptr dictionary(new DictionaryVar); dictionary->SetWithStringKey(".", PP_MakeUndefined()); dictionary->SetWithStringKey("x.y", PP_MakeUndefined()); EXPECT_TRUE(RoundTripAndCompare(dictionary->GetPPVar())); } { // Test non-string key types. They should be cast to strings. v8::Context::Scope context_scope(context_); v8::HandleScope handle_scope; const char* source = "(function() {" "return {" "1: 'foo'," "'2': 'bar'," "true: 'baz'," "false: 'qux'," "null: 'quux'," "undefined: 'oops'" "};" "})();"; v8::Handle script(v8::Script::New(v8::String::New(source))); v8::Handle object = script->Run().As(); ASSERT_FALSE(object.IsEmpty()); V8VarConverter converter; PP_Var actual; ASSERT_TRUE(converter.FromV8Value(object, context_, &actual)); ScopedPPVar release_actual(ScopedPPVar::PassRef(), actual); scoped_refptr expected(new DictionaryVar); ScopedPPVar foo(ScopedPPVar::PassRef(), StringVar::StringToPPVar("foo")); expected->SetWithStringKey("1", foo.get()); ScopedPPVar bar(ScopedPPVar::PassRef(), StringVar::StringToPPVar("bar")); expected->SetWithStringKey("2", bar.get()); ScopedPPVar baz(ScopedPPVar::PassRef(), StringVar::StringToPPVar("baz")); expected->SetWithStringKey("true", baz.get()); ScopedPPVar qux(ScopedPPVar::PassRef(), StringVar::StringToPPVar("qux")); expected->SetWithStringKey("false", qux.get()); ScopedPPVar quux(ScopedPPVar::PassRef(), StringVar::StringToPPVar("quux")); expected->SetWithStringKey("null", quux.get()); ScopedPPVar oops(ScopedPPVar::PassRef(), StringVar::StringToPPVar("oops")); expected->SetWithStringKey("undefined", oops.get()); ScopedPPVar release_expected( ScopedPPVar::PassRef(), expected->GetPPVar()); ASSERT_TRUE(TestEqual(release_expected.get(), release_actual.get())); } } } // namespace ppapi } // namespace webkit