summaryrefslogtreecommitdiffstats
path: root/chrome/browser/importer.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chrome/browser/importer.cc')
-rw-r--r--chrome/browser/importer.cc567
1 files changed, 567 insertions, 0 deletions
diff --git a/chrome/browser/importer.cc b/chrome/browser/importer.cc
new file mode 100644
index 0000000..2053527
--- /dev/null
+++ b/chrome/browser/importer.cc
@@ -0,0 +1,567 @@
+// Copyright 2008, Google Inc.
+// All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+#include "chrome/browser/importer.h"
+
+#include <map>
+
+#include "base/file_util.h"
+#include "base/gfx/image_operations.h"
+#include "base/gfx/png_encoder.h"
+#include "base/string_util.h"
+#include "chrome/browser/bookmark_bar_model.h"
+#include "chrome/browser/browser_process.h"
+#include "chrome/browser/firefox2_importer.h"
+#include "chrome/browser/firefox3_importer.h"
+#include "chrome/browser/firefox_importer_utils.h"
+#include "chrome/browser/firefox_profile_lock.h"
+#include "chrome/browser/ie_importer.h"
+#include "chrome/browser/template_url_model.h"
+#include "chrome/browser/shell_integration.h"
+#include "chrome/browser/webdata/web_data_service.h"
+#include "chrome/common/gfx/favicon_size.h"
+#include "chrome/common/l10n_util.h"
+#include "chrome/common/pref_names.h"
+#include "chrome/common/pref_service.h"
+#include "chrome/views/window.h"
+#include "webkit/glue/image_decoder.h"
+
+#include "generated_resources.h"
+
+// ProfileWriter.
+
+bool ProfileWriter::BookmarkBarModelIsLoaded() const {
+ return profile_->GetBookmarkBarModel()->IsLoaded();
+}
+
+void ProfileWriter::AddBookmarkBarModelObserver(
+ BookmarkBarModelObserver* observer) {
+ profile_->GetBookmarkBarModel()->AddObserver(observer);
+}
+
+bool ProfileWriter::TemplateURLModelIsLoaded() const {
+ return profile_->GetTemplateURLModel()->loaded();
+}
+
+void ProfileWriter::AddTemplateURLModelObserver(
+ NotificationObserver* observer) {
+ TemplateURLModel* model = profile_->GetTemplateURLModel();
+ NotificationService::current()->AddObserver(
+ observer, TEMPLATE_URL_MODEL_LOADED,
+ Source<TemplateURLModel>(model));
+ model->Load();
+}
+
+void ProfileWriter::AddPasswordForm(const PasswordForm& form) {
+ profile_->GetWebDataService(Profile::EXPLICIT_ACCESS)->AddLogin(form);
+}
+
+void ProfileWriter::AddIE7PasswordInfo(const IE7PasswordInfo& info) {
+ profile_->GetWebDataService(Profile::EXPLICIT_ACCESS)->AddIE7Login(info);
+}
+
+void ProfileWriter::AddHistoryPage(const std::vector<history::URLRow>& page) {
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS)->
+ AddPagesWithDetails(page);
+}
+
+void ProfileWriter::AddHomepage(const GURL& home_page) {
+ DCHECK(profile_);
+
+ PrefService* prefs = profile_->GetPrefs();
+ prefs->SetString(prefs::kHomePage, ASCIIToWide(home_page.spec()));
+ prefs->ScheduleSavePersistentPrefs(g_browser_process->file_thread());
+}
+
+void ProfileWriter::AddBookmarkEntry(
+ const std::vector<BookmarkEntry>& bookmark) {
+ BookmarkBarModel* model = profile_->GetBookmarkBarModel();
+ DCHECK(model->IsLoaded());
+
+ bool show_bookmark_toolbar = false;
+ std::set<BookmarkBarNode*> groups_added_to;
+ for (std::vector<BookmarkEntry>::const_iterator it = bookmark.begin();
+ it != bookmark.end(); ++it) {
+ // Don't insert this url if it exists in model or url is not valid.
+ if (model->GetNodeByURL(it->url) != NULL || !it->url.is_valid())
+ continue;
+
+ // Set up groups in BookmarkBarModel in such a way that path[i] is
+ // the subgroup of path[i-1]. Finally they construct a path in the
+ // model:
+ // path[0] \ path[1] \ ... \ path[size() - 1]
+ BookmarkBarNode* parent =
+ (it->in_toolbar ? model->GetBookmarkBarNode() : model->other_node());
+ for (std::vector<std::wstring>::const_iterator i = it->path.begin();
+ i != it->path.end(); ++i) {
+ BookmarkBarNode* child = NULL;
+ for (int index = 0; index < parent->GetChildCount(); ++index) {
+ BookmarkBarNode* node = parent->GetChild(index);
+ if ((node->GetType() == history::StarredEntry::BOOKMARK_BAR ||
+ node->GetType() == history::StarredEntry::USER_GROUP) &&
+ node->GetTitle() == *i) {
+ child = node;
+ break;
+ }
+ }
+ if (child == NULL)
+ child = model->AddGroup(parent, parent->GetChildCount(), *i);
+ parent = child;
+ }
+ groups_added_to.insert(parent);
+ model->AddURLWithCreationTime(parent, parent->GetChildCount(),
+ it->title, it->url, it->creation_time);
+
+ // If some items are put into toolbar, it looks like the user was using
+ // it in their last browser. We turn on the bookmarks toolbar.
+ if (it->in_toolbar)
+ show_bookmark_toolbar = true;
+ }
+
+ // Reset the date modified time of the groups we added to. We do this to
+ // make sure the 'recently added to' combobox in the bubble doesn't get random
+ // groups.
+ for (std::set<BookmarkBarNode*>::const_iterator i = groups_added_to.begin();
+ i != groups_added_to.end(); ++i) {
+ model->ResetDateGroupModified(*i);
+ }
+
+ if (show_bookmark_toolbar)
+ ShowBookmarkBar();
+}
+
+void ProfileWriter::AddFavicons(
+ const std::vector<history::ImportedFavIconUsage>& favicons) {
+ profile_->GetHistoryService(Profile::EXPLICIT_ACCESS)->
+ SetImportedFavicons(favicons);
+}
+
+typedef std::map<std::string, const TemplateURL*> HostPathMap;
+
+// Builds the key to use in HostPathMap for the specified TemplateURL. Returns
+// an empty string if a host+path can't be generated for the TemplateURL.
+// If an empty string is returned, it should not be added to HostPathMap.
+static std::string BuildHostPathKey(const TemplateURL* t_url) {
+ if (t_url->url() && t_url->url()->SupportsReplacement()) {
+ GURL search_url(t_url->url()->ReplaceSearchTerms(
+ *t_url, L"random string", TemplateURLRef::NO_SUGGESTIONS_AVAILABLE,
+ std::wstring()));
+ if (search_url.is_valid())
+ return search_url.host() + search_url.path();
+ }
+ return std::string();
+}
+
+// Builds a set that contains an entry of the host+path for each TemplateURL in
+// the TemplateURLModel that has a valid search url.
+static void BuildHostPathMap(const TemplateURLModel& model,
+ HostPathMap* host_path_map) {
+ std::vector<const TemplateURL*> template_urls = model.GetTemplateURLs();
+ for (size_t i = 0; i < template_urls.size(); ++i) {
+ const std::string host_path = BuildHostPathKey(template_urls[i]);
+ if (!host_path.empty()) {
+ const TemplateURL* existing_turl = (*host_path_map)[host_path];
+ if (!existing_turl || template_urls[i]->show_in_default_list()) {
+ // If there are multiple TemplateURLs with the same host+path, favor
+ // those show in the default list. This is done just in case we end
+ // up using it as the default search provider.
+ (*host_path_map)[host_path] = template_urls[i];
+ }
+ } // else case, TemplateURL doesn't have a search url, doesn't support
+ // replacement, or doesn't have valid GURL. Ignore it.
+ }
+}
+
+void ProfileWriter::AddKeywords(const std::vector<TemplateURL*>& template_urls,
+ int default_keyword_index,
+ bool unique_on_host_and_path) {
+ TemplateURLModel* model = profile_->GetTemplateURLModel();
+ HostPathMap host_path_map;
+ if (unique_on_host_and_path)
+ BuildHostPathMap(*model, &host_path_map);
+
+ for (std::vector<TemplateURL*>::const_iterator i = template_urls.begin();
+ i != template_urls.end(); ++i) {
+ TemplateURL* t_url = *i;
+ bool default_keyword =
+ default_keyword_index >= 0 &&
+ (i - template_urls.begin() == default_keyword_index);
+
+ // TemplateURLModel requires keywords to be unique. If there is already a
+ // TemplateURL with this keyword, don't import it again.
+ const TemplateURL* turl_with_keyword =
+ model->GetTemplateURLForKeyword(t_url->keyword());
+ if (turl_with_keyword != NULL) {
+ if (default_keyword)
+ model->SetDefaultSearchProvider(turl_with_keyword);
+ delete t_url;
+ continue;
+ }
+
+ // For search engines if there is already a keyword with the same
+ // host+path, we don't import it. This is done to avoid both duplicate
+ // search providers (such as two Googles, or two Yahoos) as well as making
+ // sure the search engines we provide aren't replaced by those from the
+ // imported browser.
+ if (unique_on_host_and_path &&
+ host_path_map.find(BuildHostPathKey(t_url)) != host_path_map.end()) {
+ if (default_keyword) {
+ const TemplateURL* turl_with_host_path =
+ host_path_map[BuildHostPathKey(t_url)];
+ if (turl_with_host_path)
+ model->SetDefaultSearchProvider(turl_with_host_path);
+ else
+ NOTREACHED(); // BuildHostPathMap should only insert non-null values.
+ }
+ delete t_url;
+ continue;
+ }
+ model->Add(t_url);
+ if (default_keyword)
+ model->SetDefaultSearchProvider(t_url);
+ }
+}
+
+void ProfileWriter::ShowBookmarkBar() {
+ DCHECK(profile_);
+
+ PrefService* prefs = profile_->GetPrefs();
+ // Check whether the bookmark bar is shown in current pref.
+ if (!prefs->GetBoolean(prefs::kShowBookmarkBar)) {
+ // Set the pref and notify the notification service.
+ prefs->SetBoolean(prefs::kShowBookmarkBar, true);
+ prefs->ScheduleSavePersistentPrefs(g_browser_process->file_thread());
+ Source<Profile> source(profile_);
+ NotificationService::current()->Notify(
+ NOTIFY_BOOKMARK_BAR_VISIBILITY_PREF_CHANGED, source,
+ NotificationService::NoDetails());
+ }
+}
+
+// Importer.
+
+// static
+bool Importer::ReencodeFavicon(const unsigned char* src_data, size_t src_len,
+ std::vector<unsigned char>* png_data) {
+ // Decode the favicon using WebKit's image decoder.
+ webkit_glue::ImageDecoder decoder(gfx::Size(kFavIconSize, kFavIconSize));
+ SkBitmap decoded = decoder.Decode(src_data, src_len);
+ if (decoded.empty())
+ return false; // Unable to decode.
+
+ if (decoded.width() != kFavIconSize || decoded.height() != kFavIconSize) {
+ // The bitmap is not the correct size, re-sample.
+ int new_width = decoded.width();
+ int new_height = decoded.height();
+ calc_favicon_target_size(&new_width, &new_height);
+ decoded = gfx::ImageOperations::Resize(
+ decoded, gfx::ImageOperations::RESIZE_LANCZOS3,
+ gfx::Size(new_width, new_height));
+ }
+
+ // Encode our bitmap as a PNG.
+ SkAutoLockPixels decoded_lock(decoded);
+ PNGEncoder::Encode(reinterpret_cast<unsigned char*>(decoded.getPixels()),
+ PNGEncoder::FORMAT_BGRA, decoded.width(),
+ decoded.height(), decoded.width() * 4, false, png_data);
+ return true;
+}
+
+// ImporterHost.
+
+ImporterHost::ImporterHost()
+ : observer_(NULL),
+ task_(NULL),
+ importer_(NULL),
+ file_loop_(g_browser_process->file_thread()->message_loop()),
+ waiting_for_bookmarkbar_model_(false),
+ waiting_for_template_url_model_(false),
+ is_source_readable_(true) {
+ DetectSourceProfiles();
+}
+
+ImporterHost::ImporterHost(MessageLoop* file_loop)
+ : observer_(NULL),
+ task_(NULL),
+ importer_(NULL),
+ file_loop_(file_loop),
+ waiting_for_bookmarkbar_model_(false),
+ waiting_for_template_url_model_(false),
+ is_source_readable_(true) {
+ DetectSourceProfiles();
+}
+
+ImporterHost::~ImporterHost() {
+ STLDeleteContainerPointers(source_profiles_.begin(), source_profiles_.end());
+}
+
+void ImporterHost::Loaded(BookmarkBarModel* model) {
+ model->RemoveObserver(this);
+ waiting_for_bookmarkbar_model_ = false;
+ InvokeTaskIfDone();
+}
+
+void ImporterHost::Observe(NotificationType type,
+ const NotificationSource& source,
+ const NotificationDetails& details) {
+ DCHECK(type == TEMPLATE_URL_MODEL_LOADED);
+ TemplateURLModel* model = Source<TemplateURLModel>(source).ptr();
+ NotificationService::current()->RemoveObserver(
+ this, TEMPLATE_URL_MODEL_LOADED,
+ Source<TemplateURLModel>(model));
+ waiting_for_template_url_model_ = false;
+ InvokeTaskIfDone();
+}
+
+void ImporterHost::ShowWarningDialog() {
+ ImporterLockView* view = new ImporterLockView(this);
+ ChromeViews::Window* dialog =
+ ChromeViews::Window::CreateChromeWindow(
+ GetActiveWindow(), gfx::Rect(), view, view);
+ dialog->Show();
+ view->set_dialog(dialog);
+}
+
+void ImporterHost::OnLockViewEnd(bool is_continue) {
+ if (is_continue) {
+ // User chose to continue, then we check the lock again to make
+ // sure that Firefox has been closed. Try to import the settings
+ // if successful. Otherwise, show a warning dialog.
+ firefox_lock_->Lock();
+ if (firefox_lock_->HasAcquired()) {
+ is_source_readable_ = true;
+ InvokeTaskIfDone();
+ } else {
+ ShowWarningDialog();
+ }
+ } else {
+ // User chose to skip the import process. We should delete
+ // the task and notify the ImporterHost to finish.
+ delete task_;
+ task_ = NULL;
+ importer_ = NULL;
+ ImportEnded();
+ }
+}
+
+void ImporterHost::StartImportSettings(const ProfileInfo& profile_info,
+ uint16 items,
+ ProfileWriter* writer,
+ bool first_run) {
+ // Preserves the observer and creates a task, since we do async import
+ // so that it doesn't block the UI. When the import is complete, observer
+ // will be notified.
+ writer_ = writer;
+ importer_ = CreateImporterByType(profile_info.browser_type);
+ importer_->set_first_run(first_run);
+ task_ = NewRunnableMethod(importer_, &Importer::StartImport,
+ profile_info, items, writer_.get(), this);
+
+ // We should lock the Firefox profile directory to prevent corruption.
+ if (profile_info.browser_type == FIREFOX2 ||
+ profile_info.browser_type == FIREFOX3) {
+ firefox_lock_.reset(new FirefoxProfileLock(profile_info.source_path));
+ if (!firefox_lock_->HasAcquired()) {
+ // If fail to acquire the lock, we set the source unreadable and
+ // show a warning dialog.
+ is_source_readable_ = false;
+ ShowWarningDialog();
+ }
+ }
+
+ // BookmarkBarModel should be loaded before adding IE favorites. So we
+ // observe the BookmarkBarModel if needed, and start the task after
+ // it has been loaded.
+ if ((items & FAVORITES) && !writer_->BookmarkBarModelIsLoaded()) {
+ writer_->AddBookmarkBarModelObserver(this);
+ waiting_for_bookmarkbar_model_ = true;
+ }
+
+ // Observes the TemplateURLModel if needed to import search engines from the
+ // other browser. We also check to see if we're importing bookmarks because
+ // we can import bookmark keywords from Firefox as search engines.
+ if ((items & SEARCH_ENGINES) || (items & FAVORITES)) {
+ if (!writer_->TemplateURLModelIsLoaded()) {
+ writer_->AddTemplateURLModelObserver(this);
+ waiting_for_template_url_model_ = true;
+ }
+ }
+
+ AddRef();
+ InvokeTaskIfDone();
+}
+
+void ImporterHost::Cancel() {
+ if (importer_)
+ importer_->Cancel();
+}
+
+void ImporterHost::SetObserver(Observer* observer) {
+ observer_ = observer;
+}
+
+void ImporterHost::InvokeTaskIfDone() {
+ if (waiting_for_bookmarkbar_model_ || waiting_for_template_url_model_ ||
+ !is_source_readable_)
+ return;
+ file_loop_->PostTask(FROM_HERE, task_);
+}
+
+void ImporterHost::ImportItemStarted(ImportItem item) {
+ if (observer_)
+ observer_->ImportItemStarted(item);
+}
+
+void ImporterHost::ImportItemEnded(ImportItem item) {
+ if (observer_)
+ observer_->ImportItemEnded(item);
+}
+
+void ImporterHost::ImportStarted() {
+ if (observer_)
+ observer_->ImportStarted();
+}
+
+void ImporterHost::ImportEnded() {
+ firefox_lock_.reset(); // Release the Firefox profile lock.
+ if (observer_)
+ observer_->ImportEnded();
+ Release();
+}
+
+Importer* ImporterHost::CreateImporterByType(ProfileType type) {
+ switch (type) {
+ case MS_IE:
+ return new IEImporter();
+ case FIREFOX2:
+ return new Firefox2Importer();
+ case FIREFOX3:
+ return new Firefox3Importer();
+ }
+ NOTREACHED();
+ return NULL;
+}
+
+int ImporterHost::GetAvailableProfileCount() {
+ return static_cast<int>(source_profiles_.size());
+}
+
+std::wstring ImporterHost::GetSourceProfileNameAt(int index) const {
+ DCHECK(index < static_cast<int>(source_profiles_.size()));
+ return source_profiles_[index]->description;
+}
+
+const ProfileInfo& ImporterHost::GetSourceProfileInfoAt(int index) const {
+ DCHECK(index < static_cast<int>(source_profiles_.size()));
+ return *source_profiles_[index];
+}
+
+void ImporterHost::DetectSourceProfiles() {
+ if (ShellIntegration::IsFirefoxDefaultBrowser()) {
+ DetectFirefoxProfiles();
+ DetectIEProfiles();
+ } else {
+ DetectIEProfiles();
+ DetectFirefoxProfiles();
+ }
+}
+
+void ImporterHost::DetectIEProfiles() {
+ // IE always exists and don't have multiple profiles.
+ ProfileInfo* ie = new ProfileInfo();
+ ie->description = l10n_util::GetString(IDS_IMPORT_FROM_IE);
+ ie->browser_type = MS_IE;
+ ie->source_path.clear();
+ ie->app_path.clear();
+ source_profiles_.push_back(ie);
+}
+
+void ImporterHost::DetectFirefoxProfiles() {
+ // Detects which version of Firefox is installed.
+ int version = GetCurrentFirefoxMajorVersion();
+ ProfileType firefox_type;
+ if (version == 2) {
+ firefox_type = FIREFOX2;
+ } else if (version == 3) {
+ firefox_type = FIREFOX3;
+ } else {
+ // Ignores other versions of firefox.
+ return;
+ }
+
+ std::wstring ini_file = GetProfilesINI();
+ DictionaryValue root;
+ ParseProfileINI(ini_file, &root);
+
+ std::wstring source_path;
+ for (int i = 0; ; ++i) {
+ std::wstring current_profile = L"Profile" + IntToWString(i);
+ if (!root.HasKey(current_profile)) {
+ // Profiles are continuously numbered. So we exit when we can't
+ // find the i-th one.
+ break;
+ }
+ std::wstring is_relative, path, profile_path;
+ if (root.GetString(current_profile + L".IsRelative", &is_relative) &&
+ root.GetString(current_profile + L".Path", &path)) {
+ ReplaceSubstringsAfterOffset(&path, 0, L"/", L"\\");
+
+ // IsRelative=1 means the folder path would be relative to the
+ // path of profiles.ini. IsRelative=0 refers to a custom profile
+ // location.
+ if (is_relative == L"1") {
+ profile_path = file_util::GetDirectoryFromPath(ini_file);
+ file_util::AppendToPath(&profile_path, path);
+ } else {
+ profile_path = path;
+ }
+
+ // We only import the default profile when multiple profiles exist,
+ // since the other profiles are used mostly by developers for testing.
+ // Otherwise, Profile0 will be imported.
+ std::wstring is_default;
+ if ((root.GetString(current_profile + L".Default", &is_default) &&
+ is_default == L"1") || i == 0) {
+ source_path = profile_path;
+ // We break out of the loop when we have found the default profile.
+ if (is_default == L"1")
+ break;
+ }
+ }
+ }
+
+ if (!source_path.empty()) {
+ ProfileInfo* firefox = new ProfileInfo();
+ firefox->description = l10n_util::GetString(IDS_IMPORT_FROM_FIREFOX);
+ firefox->browser_type = firefox_type;
+ firefox->source_path = source_path;
+ firefox->app_path = GetFirefoxInstallPath();
+ source_profiles_.push_back(firefox);
+ }
+}