15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/sync/test/integration/bookmarks_helper.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
71320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "base/bind.h"
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/compiler_specific.h"
91320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "base/files/file_util.h"
102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/path_service.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/rand_util.h"
122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/strings/string_number_conversions.h"
13868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "base/strings/string_util.h"
14868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "base/strings/stringprintf.h"
15868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "base/strings/utf_string_conversions.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/synchronization/waitable_event.h"
17116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch#include "base/task/cancelable_task_tracker.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/bookmarks/bookmark_model_factory.h"
191320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "chrome/browser/bookmarks/chrome_bookmark_client.h"
201320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "chrome/browser/bookmarks/chrome_bookmark_client_factory.h"
21cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)#include "chrome/browser/favicon/favicon_service.h"
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/favicon/favicon_service_factory.h"
232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/history/history_db_task.h"
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/history/history_service_factory.h"
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/profiles/profile.h"
262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/sync/glue/bookmark_change_processor.h"
27a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch#include "chrome/browser/sync/test/integration/multi_client_status_change_checker.h"
285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)#include "chrome/browser/sync/test/integration/profile_sync_service_harness.h"
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/sync/test/integration/sync_datatype_helper.h"
302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/sync/test/integration/sync_test.h"
312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/common/chrome_paths.h"
325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/test/base/ui_test_utils.h"
331320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "components/bookmarks/browser/bookmark_client.h"
34cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)#include "components/bookmarks/browser/bookmark_model.h"
35cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)#include "components/bookmarks/browser/bookmark_model_observer.h"
36cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)#include "components/bookmarks/browser/bookmark_utils.h"
37f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)#include "components/favicon_base/favicon_util.h"
381320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "components/history/core/browser/history_types.h"
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "testing/gtest/include/gtest/gtest.h"
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "third_party/skia/include/core/SkBitmap.h"
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ui/base/models/tree_node_iterator.h"
422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "ui/gfx/image/image_skia.h"
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace {
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// History task which runs all pending tasks on the history thread and
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// signals when the tasks have completed.
482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)class HistoryEmptyTask : public history::HistoryDBTask {
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  explicit HistoryEmptyTask(base::WaitableEvent* done) : done_(done) {}
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual bool RunOnDBThread(history::HistoryBackend* backend,
532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                             history::HistoryDatabase* db) OVERRIDE {
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    content::RunAllPendingInMessageLoop();
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    done_->Signal();
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual void DoneRunOnMainThread() OVERRIDE {}
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) private:
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual ~HistoryEmptyTask() {}
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  base::WaitableEvent* done_;
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Helper class used to wait for changes to take effect on the favicon of a
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// particular bookmark node in a particular bookmark model.
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)class FaviconChangeObserver : public BookmarkModelObserver {
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  FaviconChangeObserver(BookmarkModel* model, const BookmarkNode* node)
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      : model_(model),
735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        node_(node),
745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        wait_for_load_(false) {
755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    model->AddObserver(this);
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual ~FaviconChangeObserver() {
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    model_->RemoveObserver(this);
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void WaitForGetFavicon() {
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    wait_for_load_ = true;
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    content::RunMessageLoop();
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ASSERT_TRUE(node_->is_favicon_loaded());
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ASSERT_FALSE(model_->GetFavicon(node_).IsEmpty());
855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void WaitForSetFavicon() {
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    wait_for_load_ = false;
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    content::RunMessageLoop();
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  virtual void BookmarkModelLoaded(BookmarkModel* model,
915d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                                   bool ids_reassigned) OVERRIDE {}
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual void BookmarkNodeMoved(BookmarkModel* model,
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 const BookmarkNode* old_parent,
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 int old_index,
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 const BookmarkNode* new_parent,
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 int new_index) OVERRIDE {}
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual void BookmarkNodeAdded(BookmarkModel* model,
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 const BookmarkNode* parent,
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 int index) OVERRIDE {}
100010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)  virtual void BookmarkNodeRemoved(
101010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      BookmarkModel* model,
102010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      const BookmarkNode* parent,
103010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      int old_index,
104010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      const BookmarkNode* node,
105010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      const std::set<GURL>& removed_urls) OVERRIDE {}
106f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  virtual void BookmarkAllUserNodesRemoved(
107010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      BookmarkModel* model,
108010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      const std::set<GURL>& removed_urls) OVERRIDE {}
109c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual void BookmarkNodeChanged(BookmarkModel* model,
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                   const BookmarkNode* node) OVERRIDE {
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (model == model_ && node == node_)
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      model->GetFavicon(node);
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual void BookmarkNodeChildrenReordered(
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BookmarkModel* model,
1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      const BookmarkNode* node) OVERRIDE {}
1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  virtual void BookmarkNodeFaviconChanged(
1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      BookmarkModel* model,
1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      const BookmarkNode* node) OVERRIDE {
1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (model == model_ && node == node_) {
1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (!wait_for_load_ || (wait_for_load_ && node->is_favicon_loaded()))
12390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        base::MessageLoopForUI::current()->Quit();
1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) private:
1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  BookmarkModel* model_;
1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const BookmarkNode* node_;
1305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool wait_for_load_;
1315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DISALLOW_COPY_AND_ASSIGN(FaviconChangeObserver);
1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// A collection of URLs for which we have added favicons. Since loading a
1355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// favicon is an asynchronous operation and doesn't necessarily invoke a
1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// callback, this collection is used to determine if we must wait for a URL's
1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// favicon to load or not.
1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)std::set<GURL>* urls_with_favicons_ = NULL;
1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Returns the number of nodes of node type |node_type| in |model| whose
1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// titles match the string |title|.
1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int CountNodesWithTitlesMatching(BookmarkModel* model,
1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 BookmarkNode::Type node_type,
144a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)                                 const base::string16& title) {
1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ui::TreeNodeIterator<const BookmarkNode> iterator(model->root_node());
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Walk through the model tree looking for bookmark nodes of node type
1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // |node_type| whose titles match |title|.
1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int count = 0;
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (iterator.has_next()) {
1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* node = iterator.Next();
1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if ((node->type() == node_type) && (node->GetTitle() == title))
1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      ++count;
1535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return count;
1555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Checks if the favicon data in |bitmap_a| and |bitmap_b| are equivalent.
1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Returns true if they match.
159f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)bool FaviconRawBitmapsMatch(const SkBitmap& bitmap_a,
160f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                            const SkBitmap& bitmap_b) {
16158537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  if (bitmap_a.getSize() == 0U && bitmap_b.getSize() == 0U)
1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
1635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if ((bitmap_a.getSize() != bitmap_b.getSize()) ||
1645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      (bitmap_a.width() != bitmap_b.width()) ||
1655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      (bitmap_a.height() != bitmap_b.height())) {
1665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Favicon size mismatch: " << bitmap_a.getSize() << " ("
1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << bitmap_a.width() << "x" << bitmap_a.height() << ") vs. "
1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << bitmap_b.getSize() << " (" << bitmap_b.width() << "x"
1695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << bitmap_b.height() << ")";
1705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
1715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  SkAutoLockPixels bitmap_lock_a(bitmap_a);
1735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  SkAutoLockPixels bitmap_lock_b(bitmap_b);
1745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void* node_pixel_addr_a = bitmap_a.getPixels();
1755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_TRUE(node_pixel_addr_a);
1765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void* node_pixel_addr_b = bitmap_b.getPixels();
1775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_TRUE(node_pixel_addr_b);
1785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (memcmp(node_pixel_addr_a, node_pixel_addr_b, bitmap_a.getSize()) !=  0) {
1795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Favicon bitmap mismatch";
1805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
1815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
1825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;
1835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Represents a favicon image and the icon URL associated with it.
1872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)struct FaviconData {
1882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  FaviconData() {
1892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  FaviconData(const gfx::Image& favicon_image,
1922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)              const GURL& favicon_url)
1932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      : image(favicon_image),
1942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        icon_url(favicon_url) {
1952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  ~FaviconData() {
1982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  gfx::Image image;
2012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  GURL icon_url;
2022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)};
2032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Gets the favicon and icon URL associated with |node| in |model|.
2052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)FaviconData GetFaviconData(BookmarkModel* model,
2062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                           const BookmarkNode* node) {
2075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // If a favicon wasn't explicitly set for a particular URL, simply return its
2085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // blank favicon.
2095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!urls_with_favicons_ ||
2105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      urls_with_favicons_->find(node->url()) == urls_with_favicons_->end()) {
2112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return FaviconData();
2125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // If a favicon was explicitly set, we may need to wait for it to be loaded
2145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // via BookmarkModel::GetFavicon(), which is an asynchronous operation.
2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!node->is_favicon_loaded()) {
2165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    FaviconChangeObserver observer(model, node);
2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    model->GetFavicon(node);
2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    observer.WaitForGetFavicon();
2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_TRUE(node->is_favicon_loaded());
2215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_FALSE(model->GetFavicon(node).IsEmpty());
2222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return FaviconData(model->GetFavicon(node), node->icon_url());
2235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Sets the favicon for |profile| and |node|. |profile| may be
2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// |test()->verifier()|.
2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void SetFaviconImpl(Profile* profile,
2285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                    const BookmarkNode* node,
2292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                    const GURL& icon_url,
2302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                    const gfx::Image& image,
2312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                    bookmarks_helper::FaviconSource favicon_source) {
2325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile);
2335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    FaviconChangeObserver observer(model, node);
2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    FaviconService* favicon_service =
2365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        FaviconServiceFactory::GetForProfile(profile,
2375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                             Profile::EXPLICIT_ACCESS);
2382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    if (favicon_source == bookmarks_helper::FROM_UI) {
2390529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      favicon_service->SetFavicons(
2400529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch          node->url(), icon_url, favicon_base::FAVICON, image);
2412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    } else {
2422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      browser_sync::BookmarkChangeProcessor::ApplyBookmarkFavicon(
2432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)          node, profile, icon_url, image.As1xPNGBytes());
2442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    }
2455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Wait for the favicon for |node| to be invalidated.
2475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    observer.WaitForSetFavicon();
2485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Wait for the BookmarkModel to fetch the updated favicon and for the new
2495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // favicon to be sent to BookmarkChangeProcessor.
2502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    GetFaviconData(model, node);
2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Wait for all currently scheduled tasks on the history thread for all
2545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// profiles to complete and any notifications sent to the UI thread to have
2555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// finished processing.
2565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void WaitForHistoryToProcessPendingTasks() {
2575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Skip waiting for history to complete for tests without favicons.
2585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!urls_with_favicons_)
2595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
2605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::vector<Profile*> profiles_which_need_to_wait;
2625c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier())
2635c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    profiles_which_need_to_wait.push_back(
2645c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        sync_datatype_helper::test()->verifier());
2655c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  for (int i = 0; i < sync_datatype_helper::test()->num_clients(); ++i)
2665c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    profiles_which_need_to_wait.push_back(
2675c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        sync_datatype_helper::test()->GetProfile(i));
2685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (size_t i = 0; i < profiles_which_need_to_wait.size(); ++i) {
2705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Profile* profile = profiles_which_need_to_wait[i];
2715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    HistoryService* history_service =
2725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        HistoryServiceFactory::GetForProfileWithoutCreating(profile);
2735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    base::WaitableEvent done(false, false);
274116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch    base::CancelableTaskTracker task_tracker;
2755f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)    history_service->ScheduleDBTask(
2765f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)        scoped_ptr<history::HistoryDBTask>(
2775f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)            new HistoryEmptyTask(&done)),
2785f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)        &task_tracker);
2795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    done.Wait();
2805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Wait such that any notifications broadcast from one of the history threads
2825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // to the UI thread are processed.
2835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  content::RunAllPendingInMessageLoop();
2845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Checks if the favicon in |node_a| from |model_a| matches that of |node_b|
2875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// from |model_b|. Returns true if they match.
2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool FaviconsMatch(BookmarkModel* model_a,
2895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                   BookmarkModel* model_b,
2905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                   const BookmarkNode* node_a,
2915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                   const BookmarkNode* node_b) {
2922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  FaviconData favicon_data_a = GetFaviconData(model_a, node_a);
2932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  FaviconData favicon_data_b = GetFaviconData(model_b, node_b);
2945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (favicon_data_a.icon_url != favicon_data_b.icon_url)
2962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return false;
2972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  gfx::Image image_a = favicon_data_a.image;
2992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  gfx::Image image_b = favicon_data_b.image;
3002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (image_a.IsEmpty() && image_b.IsEmpty())
3025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return true;  // Two empty images are equivalent.
3032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (image_a.IsEmpty() != image_b.IsEmpty())
3052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return false;
3062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Compare only the 1x bitmaps as only those are synced.
3082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  SkBitmap bitmap_a = image_a.AsImageSkia().GetRepresentation(
30968043e1e95eeb07d5cae7aca370b26518b0867d6Torne (Richard Coles)      1.0f).sk_bitmap();
3102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  SkBitmap bitmap_b = image_b.AsImageSkia().GetRepresentation(
31168043e1e95eeb07d5cae7aca370b26518b0867d6Torne (Richard Coles)      1.0f).sk_bitmap();
312f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  return FaviconRawBitmapsMatch(bitmap_a, bitmap_b);
3135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Does a deep comparison of BookmarkNode fields in |model_a| and |model_b|.
3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Returns true if they are all equal.
3175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool NodesMatch(const BookmarkNode* node_a, const BookmarkNode* node_b) {
3185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (node_a == NULL || node_b == NULL)
3195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return node_a == node_b;
3205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (node_a->is_folder() != node_b->is_folder()) {
3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Cannot compare folder with bookmark";
3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
3235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (node_a->GetTitle() != node_b->GetTitle()) {
3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Title mismatch: " << node_a->GetTitle() << " vs. "
3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << node_b->GetTitle();
3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (node_a->url() != node_b->url()) {
3305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "URL mismatch: " << node_a->url() << " vs. "
3315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << node_b->url();
3325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
3335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (node_a->parent()->GetIndexOf(node_a) !=
3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      node_b->parent()->GetIndexOf(node_b)) {
3365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Index mismatch: "
3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << node_a->parent()->GetIndexOf(node_a) << " vs. "
3385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << node_b->parent()->GetIndexOf(node_b);
3395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
3405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
3425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3441320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci// Helper for BookmarkModelsMatch.
3451320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tuccibool NodeCantBeSynced(bookmarks::BookmarkClient* client,
3461320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci                      const BookmarkNode* node) {
3471320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // Return true to skip a node.
3481320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  return !client->CanSyncNode(node);
3491320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci}
3501320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci
3515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Checks if the hierarchies in |model_a| and |model_b| are equivalent in
3525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// terms of the data model and favicon. Returns true if they both match.
3535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Note: Some peripheral fields like creation times are allowed to mismatch.
3545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool BookmarkModelsMatch(BookmarkModel* model_a, BookmarkModel* model_b) {
3555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool ret_val = true;
3561320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  ui::TreeNodeIterator<const BookmarkNode> iterator_a(
3571320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      model_a->root_node(), base::Bind(&NodeCantBeSynced, model_a->client()));
3581320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  ui::TreeNodeIterator<const BookmarkNode> iterator_b(
3591320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      model_b->root_node(), base::Bind(&NodeCantBeSynced, model_b->client()));
3605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (iterator_a.has_next()) {
3615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* node_a = iterator_a.Next();
3625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!iterator_b.has_next()) {
3635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      LOG(ERROR) << "Models do not match.";
3645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return false;
3655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
3665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* node_b = iterator_b.Next();
3675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ret_val = ret_val && NodesMatch(node_a, node_b);
3685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (node_a->is_folder() || node_b->is_folder())
3695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      continue;
3705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ret_val = ret_val && FaviconsMatch(model_a, model_b, node_a, node_b);
3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ret_val = ret_val && (!iterator_b.has_next());
3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return ret_val;
3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Finds the node in the verifier bookmark model that corresponds to
3775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// |foreign_node| in |foreign_model| and stores its address in |result|.
3785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void FindNodeInVerifier(BookmarkModel* foreign_model,
3795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                        const BookmarkNode* foreign_node,
3805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                        const BookmarkNode** result) {
3815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Climb the tree.
3825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::stack<int> path;
3835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const BookmarkNode* walker = foreign_node;
3845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (walker != foreign_model->root_node()) {
3855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    path.push(walker->parent()->GetIndexOf(walker));
3865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    walker = walker->parent();
3875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Swing over to the other tree.
3905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  walker = bookmarks_helper::GetVerifierBookmarkModel()->root_node();
3915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Climb down.
3935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (!path.empty()) {
3945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ASSERT_TRUE(walker->is_folder());
3955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ASSERT_LT(path.top(), walker->child_count());
3965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    walker = walker->GetChild(path.top());
3975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    path.pop();
3985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ASSERT_TRUE(NodesMatch(foreign_node, walker));
4015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  *result = walker;
4025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace
4055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace bookmarks_helper {
4085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)BookmarkModel* GetBookmarkModel(int index) {
4105c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  return BookmarkModelFactory::GetForProfile(
4115c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      sync_datatype_helper::test()->GetProfile(index));
4125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* GetBookmarkBarNode(int index) {
4155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return GetBookmarkModel(index)->bookmark_bar_node();
4165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* GetOtherNode(int index) {
4195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return GetBookmarkModel(index)->other_node();
4205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* GetSyncedBookmarksNode(int index) {
4235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return GetBookmarkModel(index)->mobile_node();
4241320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci}
4251320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci
4261320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucciconst BookmarkNode* GetManagedNode(int index) {
4271320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  return ChromeBookmarkClientFactory::GetForProfile(
4281320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      sync_datatype_helper::test()->GetProfile(index))->managed_node();
4295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)BookmarkModel* GetVerifierBookmarkModel() {
4325c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  return BookmarkModelFactory::GetForProfile(
4335c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      sync_datatype_helper::test()->verifier());
4345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* AddURL(int profile,
437f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                           const std::string& title,
4385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           const GURL& url) {
4395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return AddURL(profile, GetBookmarkBarNode(profile), 0, title,  url);
4405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* AddURL(int profile,
4435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           int index,
444f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                           const std::string& title,
4455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           const GURL& url) {
4465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return AddURL(profile, GetBookmarkBarNode(profile), index, title, url);
4475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* AddURL(int profile,
4505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           const BookmarkNode* parent,
4515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           int index,
452f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                           const std::string& title,
4535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           const GURL& url) {
4540529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
455116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  if (bookmarks::GetBookmarkNodeByID(model, parent->id()) != parent) {
4565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Node " << parent->GetTitle() << " does not belong to "
4575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << "Profile " << profile;
4585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
4595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4600529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  const BookmarkNode* result =
461f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)      model->AddURL(parent, index, base::UTF8ToUTF16(title), url);
4625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!result) {
4635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Could not add bookmark " << title << " to Profile "
4645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << profile;
4655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
4665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4675c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
4685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_parent = NULL;
4690529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, parent, &v_parent);
4700529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    const BookmarkNode* v_node = GetVerifierBookmarkModel()->AddURL(
471f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)        v_parent, index, base::UTF8ToUTF16(title), url);
4725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!v_node) {
4735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      LOG(ERROR) << "Could not add bookmark " << title << " to the verifier";
4745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return NULL;
4755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    EXPECT_TRUE(NodesMatch(v_node, result));
4775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return result;
4795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* AddFolder(int profile,
482f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                              const std::string& title) {
4835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return AddFolder(profile, GetBookmarkBarNode(profile), 0, title);
4845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* AddFolder(int profile,
4875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                              int index,
488f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                              const std::string& title) {
4895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return AddFolder(profile, GetBookmarkBarNode(profile), index, title);
4905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* AddFolder(int profile,
4935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                              const BookmarkNode* parent,
4945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                              int index,
495f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                              const std::string& title) {
4960529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
497116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  if (bookmarks::GetBookmarkNodeByID(model, parent->id()) != parent) {
4985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Node " << parent->GetTitle() << " does not belong to "
4995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << "Profile " << profile;
5005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
5015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5020529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  const BookmarkNode* result =
503f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)      model->AddFolder(parent, index, base::UTF8ToUTF16(title));
5045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_TRUE(result);
5055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!result) {
5065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Could not add folder " << title << " to Profile "
5075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << profile;
5085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
5095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5105c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
5115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_parent = NULL;
5120529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, parent, &v_parent);
5135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_node = GetVerifierBookmarkModel()->AddFolder(
514f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)        v_parent, index, base::UTF8ToUTF16(title));
5155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!v_node) {
5165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      LOG(ERROR) << "Could not add folder " << title << " to the verifier";
5175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return NULL;
5185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    EXPECT_TRUE(NodesMatch(v_node, result));
5205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return result;
5225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void SetTitle(int profile,
5255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)              const BookmarkNode* node,
526f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)              const std::string& new_title) {
5270529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
528116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  ASSERT_EQ(bookmarks::GetBookmarkNodeByID(model, node->id()), node)
5295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Node " << node->GetTitle() << " does not belong to "
5305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Profile " << profile;
5315c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
5325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_node = NULL;
5330529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, node, &v_node);
534f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)    GetVerifierBookmarkModel()->SetTitle(v_node, base::UTF8ToUTF16(new_title));
5355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
536f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  model->SetTitle(node, base::UTF8ToUTF16(new_title));
5375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void SetFavicon(int profile,
5405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                const BookmarkNode* node,
5412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                const GURL& icon_url,
5422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                const gfx::Image& image,
5432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                FaviconSource favicon_source) {
5440529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
545116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  ASSERT_EQ(bookmarks::GetBookmarkNodeByID(model, node->id()), node)
5465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Node " << node->GetTitle() << " does not belong to "
5475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Profile " << profile;
5480529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  ASSERT_EQ(BookmarkNode::URL, node->type()) << "Node " << node->GetTitle()
5490529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch                                             << " must be a url.";
5505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (urls_with_favicons_ == NULL)
5515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    urls_with_favicons_ = new std::set<GURL>();
5525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  urls_with_favicons_->insert(node->url());
5535c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
5545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_node = NULL;
5550529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, node, &v_node);
5565c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    SetFaviconImpl(sync_datatype_helper::test()->verifier(),
5575c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                   v_node,
5585c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                   icon_url,
5595c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                   image,
5605c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                   favicon_source);
5615c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  }
5625c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  SetFaviconImpl(sync_datatype_helper::test()->GetProfile(profile),
5635c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                 node,
5645c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                 icon_url,
5655c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                 image,
5662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 favicon_source);
5675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* SetURL(int profile,
5705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           const BookmarkNode* node,
5715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                           const GURL& new_url) {
5720529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
573116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  if (bookmarks::GetBookmarkNodeByID(model, node->id()) != node) {
5745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Node " << node->GetTitle() << " does not belong to "
5755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << "Profile " << profile;
5765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
5775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5785c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
5795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_node = NULL;
5800529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, node, &v_node);
581868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    if (v_node->is_url())
582868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      GetVerifierBookmarkModel()->SetURL(v_node, new_url);
583868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  }
584868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  if (node->is_url())
5850529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    model->SetURL(node, new_url);
586868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  return node;
5875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
5885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void Move(int profile,
5905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          const BookmarkNode* node,
5915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          const BookmarkNode* new_parent,
5925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          int index) {
5930529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
594116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  ASSERT_EQ(bookmarks::GetBookmarkNodeByID(model, node->id()), node)
5955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Node " << node->GetTitle() << " does not belong to "
5965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Profile " << profile;
5975c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
5985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_new_parent = NULL;
5995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_node = NULL;
6000529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, new_parent, &v_new_parent);
6010529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, node, &v_node);
6025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    GetVerifierBookmarkModel()->Move(v_node, v_new_parent, index);
6035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6040529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  model->Move(node, new_parent, index);
6055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6070529e5d033099cbfc42635f6f6183833b09dff6eBen Murdochvoid Remove(int profile, const BookmarkNode* parent, int index) {
6080529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
609116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  ASSERT_EQ(bookmarks::GetBookmarkNodeByID(model, parent->id()), parent)
6105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Node " << parent->GetTitle() << " does not belong to "
6115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Profile " << profile;
6125c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
6135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_parent = NULL;
6140529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, parent, &v_parent);
6155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ASSERT_TRUE(NodesMatch(parent->GetChild(index), v_parent->GetChild(index)));
6165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    GetVerifierBookmarkModel()->Remove(v_parent, index);
6175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6180529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  model->Remove(parent, index);
6195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
621c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)void RemoveAll(int profile) {
6225c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
623c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    const BookmarkNode* root_node = GetVerifierBookmarkModel()->root_node();
624c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    for (int i = 0; i < root_node->child_count(); ++i) {
625c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      const BookmarkNode* permanent_node = root_node->GetChild(i);
626c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      for (int j = permanent_node->child_count() - 1; j >= 0; --j) {
627c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)        GetVerifierBookmarkModel()->Remove(permanent_node, j);
628c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      }
629c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    }
630c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
631f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  GetBookmarkModel(profile)->RemoveAllUserBookmarks();
632c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)}
633c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
6345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void SortChildren(int profile, const BookmarkNode* parent) {
6350529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  BookmarkModel* model = GetBookmarkModel(profile);
636116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  ASSERT_EQ(bookmarks::GetBookmarkNodeByID(model, parent->id()), parent)
6375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Node " << parent->GetTitle() << " does not belong to "
6385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Profile " << profile;
6395c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (sync_datatype_helper::test()->use_verifier()) {
6405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* v_parent = NULL;
6410529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    FindNodeInVerifier(model, parent, &v_parent);
6425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    GetVerifierBookmarkModel()->SortChildren(v_parent);
6435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6440529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  model->SortChildren(parent);
6455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ReverseChildOrder(int profile, const BookmarkNode* parent) {
648116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  ASSERT_EQ(
649116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch      bookmarks::GetBookmarkNodeByID(GetBookmarkModel(profile), parent->id()),
650116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch      parent)
6515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Node " << parent->GetTitle() << " does not belong to "
6525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      << "Profile " << profile;
6535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int child_count = parent->child_count();
6545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (child_count <= 0)
6555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
6565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (int index = 0; index < child_count; ++index) {
6575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Move(profile, parent->GetChild(index), parent, child_count - index);
6585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ModelMatchesVerifier(int profile) {
6625c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (!sync_datatype_helper::test()->use_verifier()) {
6635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    LOG(ERROR) << "Illegal to call ModelMatchesVerifier() after "
6645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)               << "DisableVerifier(). Use ModelsMatch() instead.";
6655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
6665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return BookmarkModelsMatch(GetVerifierBookmarkModel(),
6685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                             GetBookmarkModel(profile));
6695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool AllModelsMatchVerifier() {
6725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Ensure that all tasks have finished processing on the history thread
6735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // and that any notifications the history thread may have sent have been
6745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // processed before comparing models.
6755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  WaitForHistoryToProcessPendingTasks();
6765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6775c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  for (int i = 0; i < sync_datatype_helper::test()->num_clients(); ++i) {
6785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!ModelMatchesVerifier(i)) {
6795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      LOG(ERROR) << "Model " << i << " does not match the verifier.";
6805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return false;
6815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
6825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
6845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ModelsMatch(int profile_a, int profile_b) {
6875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return BookmarkModelsMatch(GetBookmarkModel(profile_a),
6885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                             GetBookmarkModel(profile_b));
6895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
6905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool AllModelsMatch() {
6925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Ensure that all tasks have finished processing on the history thread
6935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // and that any notifications the history thread may have sent have been
6945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // processed before comparing models.
6955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  WaitForHistoryToProcessPendingTasks();
6965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6975c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  for (int i = 1; i < sync_datatype_helper::test()->num_clients(); ++i) {
6985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!ModelsMatch(0, i)) {
6995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      LOG(ERROR) << "Model " << i << " does not match Model 0.";
7005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return false;
7015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
7025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
7045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
706a02191e04bc25c4935f804f2c080ae28663d096dBen Murdochnamespace {
707a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
708a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch// Helper class used in the implementation of AwaitAllModelsMatch.
709a02191e04bc25c4935f804f2c080ae28663d096dBen Murdochclass AllModelsMatchChecker : public MultiClientStatusChangeChecker {
710a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch public:
711a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  AllModelsMatchChecker();
712a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  virtual ~AllModelsMatchChecker();
713a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
714a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  virtual bool IsExitConditionSatisfied() OVERRIDE;
715a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  virtual std::string GetDebugMessage() const OVERRIDE;
716a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch};
717a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
718a02191e04bc25c4935f804f2c080ae28663d096dBen MurdochAllModelsMatchChecker::AllModelsMatchChecker()
719a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch    : MultiClientStatusChangeChecker(
720a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch        sync_datatype_helper::test()->GetSyncServices()) {}
721a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
722a02191e04bc25c4935f804f2c080ae28663d096dBen MurdochAllModelsMatchChecker::~AllModelsMatchChecker() {}
723a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
724a02191e04bc25c4935f804f2c080ae28663d096dBen Murdochbool AllModelsMatchChecker::IsExitConditionSatisfied() {
725a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  return AllModelsMatch();
726a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch}
727a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
728a02191e04bc25c4935f804f2c080ae28663d096dBen Murdochstd::string AllModelsMatchChecker::GetDebugMessage() const {
729a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  return "Waiting for matching models";
730a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch}
731a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
732a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch}  //  namespace
733a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
734a02191e04bc25c4935f804f2c080ae28663d096dBen Murdochbool AwaitAllModelsMatch() {
735a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  AllModelsMatchChecker checker;
736a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  checker.Wait();
737a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch  return !checker.TimedOut();
738a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch}
739a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
740a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch
7415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ContainsDuplicateBookmarks(int profile) {
7425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ui::TreeNodeIterator<const BookmarkNode> iterator(
7435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GetBookmarkModel(profile)->root_node());
7445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (iterator.has_next()) {
7455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const BookmarkNode* node = iterator.Next();
7465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (node->is_folder())
7475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      continue;
7485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    std::vector<const BookmarkNode*> nodes;
7495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    GetBookmarkModel(profile)->GetNodesByURL(node->url(), &nodes);
7505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    EXPECT_TRUE(nodes.size() >= 1);
7515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (std::vector<const BookmarkNode*>::const_iterator it = nodes.begin();
7525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)         it != nodes.end(); ++it) {
7535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      if (node->id() != (*it)->id() &&
7545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          node->parent() == (*it)->parent() &&
7555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          node->GetTitle() == (*it)->GetTitle()){
7565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        return true;
7575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      }
7585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
7595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
7615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool HasNodeWithURL(int profile, const GURL& url) {
7645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::vector<const BookmarkNode*> nodes;
7655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  GetBookmarkModel(profile)->GetNodesByURL(url, &nodes);
7665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return !nodes.empty();
7675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const BookmarkNode* GetUniqueNodeByURL(int profile, const GURL& url) {
7705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::vector<const BookmarkNode*> nodes;
7715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  GetBookmarkModel(profile)->GetNodesByURL(url, &nodes);
7725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  EXPECT_EQ(1U, nodes.size());
7735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (nodes.empty())
7745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return NULL;
7755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return nodes[0];
7765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
778f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)int CountBookmarksWithTitlesMatching(int profile, const std::string& title) {
7795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return CountNodesWithTitlesMatching(GetBookmarkModel(profile),
7805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                      BookmarkNode::URL,
781f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                                      base::UTF8ToUTF16(title));
7825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
784f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)int CountFoldersWithTitlesMatching(int profile, const std::string& title) {
7855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return CountNodesWithTitlesMatching(GetBookmarkModel(profile),
7865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                      BookmarkNode::FOLDER,
787f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)                                      base::UTF8ToUTF16(title));
7885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
7895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)gfx::Image CreateFavicon(SkColor color) {
7912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const int dip_width = 16;
7922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const int dip_height = 16;
7936d86b77056ed63eb6871182f42a9fd5f07550f90Torne (Richard Coles)  std::vector<float> favicon_scales = favicon_base::GetFaviconScales();
7942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  gfx::ImageSkia favicon;
7956d86b77056ed63eb6871182f42a9fd5f07550f90Torne (Richard Coles)  for (size_t i = 0; i < favicon_scales.size(); ++i) {
7966d86b77056ed63eb6871182f42a9fd5f07550f90Torne (Richard Coles)    float scale = favicon_scales[i];
7972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    int pixel_width = dip_width * scale;
7982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    int pixel_height = dip_height * scale;
7992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    SkBitmap bmp;
800116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch    bmp.allocN32Pixels(pixel_width, pixel_height);
8012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    bmp.eraseColor(color);
802f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)    favicon.AddRepresentation(gfx::ImageSkiaRep(bmp, scale));
8032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
8042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return gfx::Image(favicon);
8052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
8062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
8072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)gfx::Image Create1xFaviconFromPNGFile(const std::string& path) {
8082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const char* kPNGExtension = ".png";
8092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (!EndsWith(path, kPNGExtension, false))
8102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return gfx::Image();
8112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
8122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  base::FilePath full_path;
8132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (!PathService::Get(chrome::DIR_TEST_DATA, &full_path))
8142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return gfx::Image();
8152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
8162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  full_path = full_path.AppendASCII("sync").AppendASCII(path);
8172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  std::string contents;
81858537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  base::ReadFileToString(full_path, &contents);
8192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return gfx::Image::CreateFrom1xPNGBytes(
8205d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::RefCountedString::TakeString(&contents));
8215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)std::string IndexedURL(int i) {
8242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return base::StringPrintf("http://www.host.ext:1234/path/filename/%d", i);
8255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
827f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)std::string IndexedURLTitle(int i) {
828f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  return base::StringPrintf("URL Title %d", i);
8295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
831f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)std::string IndexedFolderName(int i) {
832f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  return base::StringPrintf("Folder Name %d", i);
8335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
835f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)std::string IndexedSubfolderName(int i) {
836f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  return base::StringPrintf("Subfolder Name %d", i);
8375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
839f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)std::string IndexedSubsubfolderName(int i) {
840f8ee788a64d60abd8f2d742a5fdedde054ecd910Torne (Richard Coles)  return base::StringPrintf("Subsubfolder Name %d", i);
8415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace bookmarks_helper
844