// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/browser/importer/in_process_importer_bridge.h" #include #include "base/bind.h" #include "base/files/file_util.h" #include "base/macros.h" #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "build/build_config.h" #include "chrome/browser/importer/external_process_importer_host.h" #include "chrome/browser/search_engines/ui_thread_search_terms_data.h" #include "chrome/common/importer/imported_bookmark_entry.h" #include "chrome/common/importer/importer_autofill_form_data_entry.h" #include "components/autofill/core/browser/webdata/autofill_entry.h" #include "components/autofill/core/common/password_form.h" #include "components/favicon_base/favicon_usage_data.h" #include "components/search_engines/template_url.h" #include "components/search_engines/template_url_parser.h" #include "components/search_engines/template_url_prepopulate_data.h" #include "content/public/browser/browser_thread.h" #include "ui/base/l10n/l10n_util.h" #if defined(OS_WIN) #include "components/os_crypt/ie7_password_win.h" #endif #include namespace { history::URLRows ConvertImporterURLRowsToHistoryURLRows( const std::vector& rows) { history::URLRows converted; converted.reserve(rows.size()); for (std::vector::const_iterator it = rows.begin(); it != rows.end(); ++it) { history::URLRow row(it->url); row.set_title(it->title); row.set_visit_count(it->visit_count); row.set_typed_count(it->typed_count); row.set_last_visit(it->last_visit); row.set_hidden(it->hidden); converted.push_back(row); } return converted; } history::VisitSource ConvertImporterVisitSourceToHistoryVisitSource( importer::VisitSource visit_source) { switch (visit_source) { case importer::VISIT_SOURCE_BROWSED: return history::SOURCE_BROWSED; case importer::VISIT_SOURCE_FIREFOX_IMPORTED: return history::SOURCE_FIREFOX_IMPORTED; case importer::VISIT_SOURCE_IE_IMPORTED: return history::SOURCE_IE_IMPORTED; case importer::VISIT_SOURCE_SAFARI_IMPORTED: return history::SOURCE_SAFARI_IMPORTED; } NOTREACHED(); return history::SOURCE_SYNCED; } } // namespace using content::BrowserThread; namespace { // FirefoxURLParameterFilter is used to remove parameter mentioning Firefox from // the search URL when importing search engines. class FirefoxURLParameterFilter : public TemplateURLParser::ParameterFilter { public: FirefoxURLParameterFilter() {} ~FirefoxURLParameterFilter() override {} // TemplateURLParser::ParameterFilter method. bool KeepParameter(const std::string& key, const std::string& value) override { std::string low_value = base::ToLowerASCII(value); if (low_value.find("mozilla") != std::string::npos || low_value.find("firefox") != std::string::npos || low_value.find("moz:") != std::string::npos) { return false; } return true; } private: DISALLOW_COPY_AND_ASSIGN(FirefoxURLParameterFilter); }; // Attempts to create a TemplateURL from the provided data. |title| is optional. // If TemplateURL creation fails, returns NULL. // This function transfers ownership of the created TemplateURL to the caller. TemplateURL* CreateTemplateURL(const base::string16& url, const base::string16& keyword, const base::string16& title) { if (url.empty() || keyword.empty()) return NULL; TemplateURLData data; data.SetKeyword(keyword); // We set short name by using the title if it exists. // Otherwise, we use the shortcut. data.SetShortName(title.empty() ? keyword : title); data.SetURL(TemplateURLRef::DisplayURLToURLRef(url)); return new TemplateURL(data); } // Parses the OpenSearch XML files in |xml_files| and populates |search_engines| // with the resulting TemplateURLs. void ParseSearchEnginesFromFirefoxXMLData( const std::vector& xml_data, std::vector* search_engines) { DCHECK(search_engines); typedef std::map SearchEnginesMap; SearchEnginesMap search_engine_for_url; FirefoxURLParameterFilter param_filter; // The first XML file represents the default search engine in Firefox 3, so we // need to keep it on top of the list. SearchEnginesMap::const_iterator default_turl = search_engine_for_url.end(); for (std::vector::const_iterator xml_iter = xml_data.begin(); xml_iter != xml_data.end(); ++xml_iter) { TemplateURL* template_url = TemplateURLParser::Parse( UIThreadSearchTermsData(NULL), true, xml_iter->data(), xml_iter->length(), ¶m_filter); if (template_url) { SearchEnginesMap::iterator iter = search_engine_for_url.find(template_url->url()); if (iter == search_engine_for_url.end()) { iter = search_engine_for_url.insert( std::make_pair(template_url->url(), template_url)).first; } else { // We have already found a search engine with the same URL. We give // priority to the latest one found, as GetSearchEnginesXMLFiles() // returns a vector with first Firefox default search engines and then // the user's ones. We want to give priority to the user ones. delete iter->second; iter->second = template_url; } if (default_turl == search_engine_for_url.end()) default_turl = iter; } } // Put the results in the |search_engines| vector. for (SearchEnginesMap::iterator t_iter = search_engine_for_url.begin(); t_iter != search_engine_for_url.end(); ++t_iter) { if (t_iter == default_turl) search_engines->insert(search_engines->begin(), default_turl->second); else search_engines->push_back(t_iter->second); } } } // namespace InProcessImporterBridge::InProcessImporterBridge( ProfileWriter* writer, base::WeakPtr host) : writer_(writer), host_(host) { } void InProcessImporterBridge::AddBookmarks( const std::vector& bookmarks, const base::string16& first_folder_name) { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddBookmarks, writer_, bookmarks, first_folder_name)); } void InProcessImporterBridge::AddHomePage(const GURL& home_page) { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddHomepage, writer_, home_page)); } #if defined(OS_WIN) void InProcessImporterBridge::AddIE7PasswordInfo( const importer::ImporterIE7PasswordInfo& password_info) { IE7PasswordInfo ie7_password_info; ie7_password_info.url_hash = password_info.url_hash; ie7_password_info.encrypted_data = password_info.encrypted_data; ie7_password_info.date_created = password_info.date_created; BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddIE7PasswordInfo, writer_, ie7_password_info)); } #endif // OS_WIN void InProcessImporterBridge::SetFavicons( const favicon_base::FaviconUsageDataList& favicons) { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddFavicons, writer_, favicons)); } void InProcessImporterBridge::SetHistoryItems( const std::vector& rows, importer::VisitSource visit_source) { history::URLRows converted_rows = ConvertImporterURLRowsToHistoryURLRows(rows); history::VisitSource converted_visit_source = ConvertImporterVisitSourceToHistoryVisitSource(visit_source); BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddHistoryPage, writer_, converted_rows, converted_visit_source)); } void InProcessImporterBridge::SetKeywords( const std::vector& search_engines, bool unique_on_host_and_path) { ScopedVector owned_template_urls; for (const auto& search_engine : search_engines) { TemplateURL* owned_template_url = CreateTemplateURL(search_engine.url, search_engine.keyword, search_engine.display_name); if (owned_template_url) owned_template_urls.push_back(owned_template_url); } BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddKeywords, writer_, base::Passed(&owned_template_urls), unique_on_host_and_path)); } void InProcessImporterBridge::SetFirefoxSearchEnginesXMLData( const std::vector& search_engine_data) { std::vector search_engines; ParseSearchEnginesFromFirefoxXMLData(search_engine_data, &search_engines); ScopedVector owned_template_urls; std::copy(search_engines.begin(), search_engines.end(), std::back_inserter(owned_template_urls)); BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddKeywords, writer_, base::Passed(&owned_template_urls), true)); } void InProcessImporterBridge::SetPasswordForm( const autofill::PasswordForm& form) { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddPasswordForm, writer_, form)); } void InProcessImporterBridge::SetAutofillFormData( const std::vector& entries) { std::vector autofill_entries; for (size_t i = 0; i < entries.size(); ++i) { autofill_entries.push_back(autofill::AutofillEntry( autofill::AutofillKey(entries[i].name, entries[i].value), entries[i].first_used, entries[i].last_used)); } BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, base::Bind(&ProfileWriter::AddAutofillFormDataEntries, writer_, autofill_entries)); } void InProcessImporterBridge::NotifyStarted() { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ExternalProcessImporterHost::NotifyImportStarted, host_)); } void InProcessImporterBridge::NotifyItemStarted(importer::ImportItem item) { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ExternalProcessImporterHost::NotifyImportItemStarted, host_, item)); } void InProcessImporterBridge::NotifyItemEnded(importer::ImportItem item) { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ExternalProcessImporterHost::NotifyImportItemEnded, host_, item)); } void InProcessImporterBridge::NotifyEnded() { BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&ExternalProcessImporterHost::NotifyImportEnded, host_)); } base::string16 InProcessImporterBridge::GetLocalizedString(int message_id) { return l10n_util::GetStringUTF16(message_id); } InProcessImporterBridge::~InProcessImporterBridge() {}