// Copyright (c) 2011 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 "base/message_loop.h" #include "base/path_service.h" #include "chrome/browser/extensions/image_loading_tracker.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/extensions/extension.h" #include "chrome/common/extensions/extension_icon_set.h" #include "chrome/common/extensions/extension_resource.h" #include "content/browser/browser_thread.h" #include "content/common/json_value_serializer.h" #include "content/common/notification_service.h" #include "content/common/notification_type.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/skia/include/core/SkBitmap.h" #include "ui/gfx/size.h" class ImageLoadingTrackerTest : public testing::Test, public ImageLoadingTracker::Observer { public: ImageLoadingTrackerTest() : image_loaded_count_(0), quit_in_image_loaded_(false), ui_thread_(BrowserThread::UI, &ui_loop_), file_thread_(BrowserThread::FILE), io_thread_(BrowserThread::IO) { } virtual void OnImageLoaded(SkBitmap* image, const ExtensionResource& resource, int index) { image_loaded_count_++; if (quit_in_image_loaded_) MessageLoop::current()->Quit(); if (image) image_ = *image; else image_.reset(); } void WaitForImageLoad() { quit_in_image_loaded_ = true; MessageLoop::current()->Run(); quit_in_image_loaded_ = false; } int image_loaded_count() { int result = image_loaded_count_; image_loaded_count_ = 0; return result; } scoped_refptr<Extension> CreateExtension() { // Create and load an extension. FilePath test_file; if (!PathService::Get(chrome::DIR_TEST_DATA, &test_file)) { EXPECT_FALSE(true); return NULL; } test_file = test_file.AppendASCII("extensions") .AppendASCII("image_loading_tracker"); int error_code = 0; std::string error; JSONFileValueSerializer serializer(test_file.AppendASCII("app.json")); scoped_ptr<DictionaryValue> valid_value( static_cast<DictionaryValue*>(serializer.Deserialize(&error_code, &error))); EXPECT_EQ(0, error_code) << error; if (error_code != 0) return NULL; EXPECT_TRUE(valid_value.get()); if (!valid_value.get()) return NULL; return Extension::Create(test_file, Extension::INVALID, *valid_value, Extension::STRICT_ERROR_CHECKS, &error); } SkBitmap image_; private: virtual void SetUp() { file_thread_.Start(); io_thread_.Start(); } int image_loaded_count_; bool quit_in_image_loaded_; MessageLoop ui_loop_; BrowserThread ui_thread_; BrowserThread file_thread_; BrowserThread io_thread_; }; // Tests asking ImageLoadingTracker to cache pushes the result to the Extension. TEST_F(ImageLoadingTrackerTest, Cache) { scoped_refptr<Extension> extension(CreateExtension()); ASSERT_TRUE(extension.get() != NULL); ExtensionResource image_resource = extension->GetIconResource(Extension::EXTENSION_ICON_SMALLISH, ExtensionIconSet::MATCH_EXACTLY); gfx::Size max_size(Extension::EXTENSION_ICON_SMALLISH, Extension::EXTENSION_ICON_SMALLISH); ImageLoadingTracker loader(static_cast<ImageLoadingTracker::Observer*>(this)); loader.LoadImage(extension.get(), image_resource, max_size, ImageLoadingTracker::CACHE); // The image isn't cached, so we should not have received notification. EXPECT_EQ(0, image_loaded_count()); WaitForImageLoad(); // We should have gotten the image. EXPECT_EQ(1, image_loaded_count()); // Check that the image was loaded. EXPECT_EQ(Extension::EXTENSION_ICON_SMALLISH, image_.width()); // The image should be cached in the Extension. EXPECT_TRUE(extension->HasCachedImage(image_resource, max_size)); // Make sure the image is in the extension. EXPECT_EQ(Extension::EXTENSION_ICON_SMALLISH, extension->GetCachedImage(image_resource, max_size).width()); // Ask the tracker for the image again, this should call us back immediately. loader.LoadImage(extension.get(), image_resource, max_size, ImageLoadingTracker::CACHE); // We should have gotten the image. EXPECT_EQ(1, image_loaded_count()); // Check that the image was loaded. EXPECT_EQ(Extension::EXTENSION_ICON_SMALLISH, image_.width()); } // Tests deleting an extension while waiting for the image to load doesn't cause // problems. TEST_F(ImageLoadingTrackerTest, DeleteExtensionWhileWaitingForCache) { scoped_refptr<Extension> extension(CreateExtension()); ASSERT_TRUE(extension.get() != NULL); ExtensionResource image_resource = extension->GetIconResource(Extension::EXTENSION_ICON_SMALLISH, ExtensionIconSet::MATCH_EXACTLY); ImageLoadingTracker loader(static_cast<ImageLoadingTracker::Observer*>(this)); loader.LoadImage(extension.get(), image_resource, gfx::Size(Extension::EXTENSION_ICON_SMALLISH, Extension::EXTENSION_ICON_SMALLISH), ImageLoadingTracker::CACHE); // The image isn't cached, so we should not have received notification. EXPECT_EQ(0, image_loaded_count()); // Send out notification the extension was uninstalled. UnloadedExtensionInfo details(extension.get(), UnloadedExtensionInfo::UNINSTALL); NotificationService::current()->Notify( NotificationType::EXTENSION_UNLOADED, NotificationService::AllSources(), Details<UnloadedExtensionInfo>(&details)); // Chuck the extension, that way if anyone tries to access it we should crash // or get valgrind errors. extension = NULL; WaitForImageLoad(); // Even though we deleted the extension, we should still get the image. // We should still have gotten the image. EXPECT_EQ(1, image_loaded_count()); // Check that the image was loaded. EXPECT_EQ(Extension::EXTENSION_ICON_SMALLISH, image_.width()); }