/* * Copyright (C) 2012 Igalia S.L. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2,1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include "config.h" #include "WebKitTestServer.h" #include "WebViewTest.h" #include #include #include static WebKitTestServer* kServer; static char* kTempDirectory; class FaviconDatabaseTest: public WebViewTest { public: MAKE_GLIB_TEST_FIXTURE(FaviconDatabaseTest); FaviconDatabaseTest() : m_webContext(webkit_web_context_get_default()) , m_favicon(0) , m_error(0) , m_iconReadySignalReceived(false) , m_faviconNotificationReceived(false) { WebKitFaviconDatabase* database = webkit_web_context_get_favicon_database(m_webContext); g_signal_connect(database, "favicon-ready", G_CALLBACK(iconReadyCallback), this); } ~FaviconDatabaseTest() { if (m_favicon) cairo_surface_destroy(m_favicon); WebKitFaviconDatabase* database = webkit_web_context_get_favicon_database(m_webContext); g_signal_handlers_disconnect_matched(database, G_SIGNAL_MATCH_DATA, 0, 0, 0, 0, this); } static void iconReadyCallback(WebKitFaviconDatabase* database, const char* pageURI, FaviconDatabaseTest* test) { g_assert_cmpstr(webkit_web_view_get_uri(test->m_webView), ==, pageURI); test->m_iconReadySignalReceived = true; } static void faviconChangedCallback(WebKitWebView* webView, GParamSpec* pspec, gpointer data) { FaviconDatabaseTest* test = static_cast(data); g_assert(test->m_webView == webView); test->m_faviconNotificationReceived = true; test->quitMainLoop(); } static void getFaviconCallback(GObject* sourceObject, GAsyncResult* result, void* data) { FaviconDatabaseTest* test = static_cast(data); WebKitFaviconDatabase* database = webkit_web_context_get_favicon_database(test->m_webContext); test->m_favicon = webkit_favicon_database_get_favicon_finish(database, result, &test->m_error.outPtr()); test->quitMainLoop(); } void waitUntilFaviconChanged() { m_faviconNotificationReceived = false; unsigned long handlerID = g_signal_connect(m_webView, "notify::favicon", G_CALLBACK(faviconChangedCallback), this); g_main_loop_run(m_mainLoop); g_signal_handler_disconnect(m_webView, handlerID); } void getFaviconForPageURIAndWaitUntilReady(const char* pageURI) { m_error.clear(); if (m_favicon) { cairo_surface_destroy(m_favicon); m_favicon = 0; } WebKitFaviconDatabase* database = webkit_web_context_get_favicon_database(m_webContext); webkit_favicon_database_get_favicon(database, pageURI, 0, getFaviconCallback, this); g_main_loop_run(m_mainLoop); } WebKitWebContext* m_webContext; cairo_surface_t* m_favicon; GOwnPtr m_error; bool m_iconReadySignalReceived; bool m_faviconNotificationReceived; }; static void serverCallback(SoupServer* server, SoupMessage* message, const char* path, GHashTable* query, SoupClientContext* context, void* data) { if (message->method != SOUP_METHOD_GET) { soup_message_set_status(message, SOUP_STATUS_NOT_IMPLEMENTED); return; } char* contents; gsize length; if (g_str_equal(path, "/favicon.ico")) { GOwnPtr pathToFavicon(g_build_filename(Test::getWebKit1TestResoucesDir().data(), "blank.ico", NULL)); g_file_get_contents(pathToFavicon.get(), &contents, &length, 0); } else if (g_str_equal(path, "/nofavicon/favicon.ico")) { soup_message_set_status(message, SOUP_STATUS_NOT_FOUND); soup_message_body_complete(message->response_body); return; } else { contents = g_strdup("test"); length = strlen(contents); } soup_message_set_status(message, SOUP_STATUS_OK); soup_message_body_append(message->response_body, SOUP_MEMORY_TAKE, contents, length); soup_message_body_complete(message->response_body); } static void testNotInitialized(FaviconDatabaseTest* test, gconstpointer) { // Try to retrieve a valid favicon from a not initialized database. test->getFaviconForPageURIAndWaitUntilReady(kServer->getURIForPath("/").data()); g_assert(!test->m_favicon); g_assert(test->m_error); g_assert_cmpint(test->m_error->code, ==, WEBKIT_FAVICON_DATABASE_ERROR_NOT_INITIALIZED); } static void testSetDirectory(FaviconDatabaseTest* test, gconstpointer) { webkit_web_context_set_favicon_database_directory(test->m_webContext, kTempDirectory); g_assert_cmpstr(kTempDirectory, ==, webkit_web_context_get_favicon_database_directory(test->m_webContext)); } static void testClearDatabase(FaviconDatabaseTest* test, gconstpointer) { WebKitFaviconDatabase* database = webkit_web_context_get_favicon_database(test->m_webContext); webkit_favicon_database_clear(database); GOwnPtr iconURI(webkit_favicon_database_get_favicon_uri(database, kServer->getURIForPath("/").data())); g_assert(!iconURI); } static void testGetFavicon(FaviconDatabaseTest* test, gconstpointer) { // We need to load the page first to ensure the icon data will be // in the database in case there's an associated favicon. test->loadURI(kServer->getURIForPath("/").data()); test->waitUntilFaviconChanged(); g_assert(test->m_iconReadySignalReceived); // Check the API retrieving a valid favicon. test->m_iconReadySignalReceived = false; test->getFaviconForPageURIAndWaitUntilReady(kServer->getURIForPath("/").data()); g_assert(test->m_favicon); g_assert(!test->m_error); g_assert(!test->m_iconReadySignalReceived); // Check that width and height match those from blank.ico (16x16 favicon). g_assert_cmpint(cairo_image_surface_get_width(test->m_favicon), ==, 16); g_assert_cmpint(cairo_image_surface_get_height(test->m_favicon), ==, 16); // Check the API retrieving an invalid favicon. test->loadURI(kServer->getURIForPath("/nofavicon").data()); test->waitUntilFaviconChanged(); g_assert(!test->m_iconReadySignalReceived); test->getFaviconForPageURIAndWaitUntilReady(kServer->getURIForPath("/nofavicon/").data()); g_assert(!test->m_favicon); g_assert(test->m_error); g_assert(!test->m_iconReadySignalReceived); } static void testGetFaviconURI(FaviconDatabaseTest* test, gconstpointer) { WebKitFaviconDatabase* database = webkit_web_context_get_favicon_database(test->m_webContext); const char* baseURI = kServer->getURIForPath("/").data(); GOwnPtr iconURI(webkit_favicon_database_get_favicon_uri(database, baseURI)); g_assert_cmpstr(iconURI.get(), ==, kServer->getURIForPath("/favicon.ico").data()); } static void testWebViewFavicon(FaviconDatabaseTest* test, gconstpointer) { cairo_surface_t* iconFromWebView = webkit_web_view_get_favicon(test->m_webView); g_assert(!iconFromWebView); test->loadURI(kServer->getURIForPath("/").data()); test->waitUntilFaviconChanged(); g_assert(test->m_faviconNotificationReceived); iconFromWebView = webkit_web_view_get_favicon(test->m_webView); g_assert(iconFromWebView); g_assert_cmpuint(cairo_image_surface_get_width(iconFromWebView), ==, 16); g_assert_cmpuint(cairo_image_surface_get_height(iconFromWebView), ==, 16); } void beforeAll() { // Start a soup server for testing. kServer = new WebKitTestServer(); kServer->run(serverCallback); kTempDirectory = g_dir_make_tmp("WebKit2Tests-XXXXXX", 0); g_assert(kTempDirectory); // Add tests to the suite. FaviconDatabaseTest::add("WebKitFaviconDatabase", "not-initialized", testNotInitialized); FaviconDatabaseTest::add("WebKitFaviconDatabase", "set-directory", testSetDirectory); FaviconDatabaseTest::add("WebKitFaviconDatabase", "get-favicon", testGetFavicon); FaviconDatabaseTest::add("WebKitFaviconDatabase", "get-favicon-uri", testGetFaviconURI); FaviconDatabaseTest::add("WebKitFaviconDatabase", "clear-database", testClearDatabase); FaviconDatabaseTest::add("WebKitWebView", "favicon", testWebViewFavicon); } static void webkitFaviconDatabaseFinalizedCallback(gpointer, GObject*) { if (!g_file_test(kTempDirectory, G_FILE_TEST_IS_DIR)) return; GOwnPtr filename(g_build_filename(kTempDirectory, "WebpageIcons.db", NULL)); g_unlink(filename.get()); g_rmdir(kTempDirectory); } void afterAll() { delete kServer; // Delete the temporary files after the IconDatabase has been // closed, that is, once WebKitFaviconDatabase is being destroyed. WebKitFaviconDatabase* database = webkit_web_context_get_favicon_database(webkit_web_context_get_default()); g_object_weak_ref(G_OBJECT(database), webkitFaviconDatabaseFinalizedCallback, 0); }