// 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/spellchecker/spellcheck_hunspell_dictionary.h" #include "base/file_util.h" #include "base/files/memory_mapped_file.h" #include "base/logging.h" #include "base/message_loop/message_loop.h" #include "base/path_service.h" #include "chrome/browser/spellchecker/spellcheck_platform_mac.h" #include "chrome/browser/spellchecker/spellcheck_service.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/spellcheck_common.h" #include "chrome/common/spellcheck_messages.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/render_process_host.h" #include "net/base/load_flags.h" #include "net/url_request/url_fetcher.h" #include "net/url_request/url_request_context_getter.h" #include "third_party/hunspell/google/bdict.h" #include "url/gurl.h" using content::BrowserThread; namespace { // Close the file. void CloseDictionary(base::File file) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); file.Close(); } // Saves |data| to file at |path|. Returns true on successful save, otherwise // returns false. bool SaveDictionaryData(scoped_ptr data, const base::FilePath& path) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); size_t bytes_written = base::WriteFile(path, data->data(), data->length()); if (bytes_written != data->length()) { bool success = false; #if defined(OS_WIN) base::FilePath dict_dir; PathService::Get(chrome::DIR_USER_DATA, &dict_dir); base::FilePath fallback_file_path = dict_dir.Append(path.BaseName()); bytes_written = base::WriteFile(fallback_file_path, data->data(), data->length()); if (bytes_written == data->length()) success = true; #endif if (!success) { base::DeleteFile(path, false); return false; } } return true; } } // namespace SpellcheckHunspellDictionary::DictionaryFile::DictionaryFile() { } SpellcheckHunspellDictionary::DictionaryFile::~DictionaryFile() { if (file.IsValid()) { BrowserThread::PostTask( BrowserThread::FILE, FROM_HERE, base::Bind(&CloseDictionary, Passed(&file))); } } SpellcheckHunspellDictionary::DictionaryFile::DictionaryFile(RValue other) : path(other.object->path), file(other.object->file.Pass()) { } SpellcheckHunspellDictionary::DictionaryFile& SpellcheckHunspellDictionary::DictionaryFile::operator=(RValue other) { if (this != other.object) { path = other.object->path; file = other.object->file.Pass(); } return *this; } SpellcheckHunspellDictionary::SpellcheckHunspellDictionary( const std::string& language, net::URLRequestContextGetter* request_context_getter, SpellcheckService* spellcheck_service) : language_(language), use_platform_spellchecker_(false), request_context_getter_(request_context_getter), spellcheck_service_(spellcheck_service), download_status_(DOWNLOAD_NONE), weak_ptr_factory_(this) { } SpellcheckHunspellDictionary::~SpellcheckHunspellDictionary() { } void SpellcheckHunspellDictionary::Load() { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); #if defined(OS_MACOSX) if (spellcheck_mac::SpellCheckerAvailable() && spellcheck_mac::PlatformSupportsLanguage(language_)) { use_platform_spellchecker_ = true; spellcheck_mac::SetLanguage(language_); base::MessageLoop::current()->PostTask(FROM_HERE, base::Bind( &SpellcheckHunspellDictionary::InformListenersOfInitialization, weak_ptr_factory_.GetWeakPtr())); return; } #endif // OS_MACOSX BrowserThread::PostTaskAndReplyWithResult( BrowserThread::FILE, FROM_HERE, base::Bind(&InitializeDictionaryLocation, language_), base::Bind( &SpellcheckHunspellDictionary::InitializeDictionaryLocationComplete, weak_ptr_factory_.GetWeakPtr())); } void SpellcheckHunspellDictionary::RetryDownloadDictionary( net::URLRequestContextGetter* request_context_getter) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); request_context_getter_ = request_context_getter; DownloadDictionary(GetDictionaryURL()); } bool SpellcheckHunspellDictionary::IsReady() const { return GetDictionaryFile().IsValid() || IsUsingPlatformChecker(); } const base::File& SpellcheckHunspellDictionary::GetDictionaryFile() const { return dictionary_file_.file; } const std::string& SpellcheckHunspellDictionary::GetLanguage() const { return language_; } bool SpellcheckHunspellDictionary::IsUsingPlatformChecker() const { return use_platform_spellchecker_; } void SpellcheckHunspellDictionary::AddObserver(Observer* observer) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); observers_.AddObserver(observer); } void SpellcheckHunspellDictionary::RemoveObserver(Observer* observer) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); observers_.RemoveObserver(observer); } bool SpellcheckHunspellDictionary::IsDownloadInProgress() { return download_status_ == DOWNLOAD_IN_PROGRESS; } bool SpellcheckHunspellDictionary::IsDownloadFailure() { return download_status_ == DOWNLOAD_FAILED; } void SpellcheckHunspellDictionary::OnURLFetchComplete( const net::URLFetcher* source) { DCHECK(source); DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); scoped_ptr fetcher_destructor(fetcher_.release()); if ((source->GetResponseCode() / 100) != 2) { // Initialize will not try to download the file a second time. InformListenersOfDownloadFailure(); return; } // Basic sanity check on the dictionary. There's a small chance of 200 status // code for a body that represents some form of failure. scoped_ptr data(new std::string); source->GetResponseAsString(data.get()); if (data->size() < 4 || data->compare(0, 4, "BDic") != 0) { InformListenersOfDownloadFailure(); return; } // To prevent corrupted dictionary data from causing a renderer crash, scan // the dictionary data and verify it is sane before save it to a file. // TODO(rlp): Adding metrics to RecordDictionaryCorruptionStats if (!hunspell::BDict::Verify(data->data(), data->size())) { // Let PostTaskAndReply caller send to InformListenersOfInitialization // through SaveDictionaryDataComplete(). SaveDictionaryDataComplete(false); return; } BrowserThread::PostTaskAndReplyWithResult( BrowserThread::FILE, FROM_HERE, base::Bind(&SaveDictionaryData, base::Passed(&data), dictionary_file_.path), base::Bind(&SpellcheckHunspellDictionary::SaveDictionaryDataComplete, weak_ptr_factory_.GetWeakPtr())); } GURL SpellcheckHunspellDictionary::GetDictionaryURL() { static const char kDownloadServerUrl[] = "http://cache.pack.google.com/edgedl/chrome/dict/"; std::string bdict_file = dictionary_file_.path.BaseName().MaybeAsASCII(); DCHECK(!bdict_file.empty()); return GURL(std::string(kDownloadServerUrl) + StringToLowerASCII(bdict_file)); } void SpellcheckHunspellDictionary::DownloadDictionary(GURL url) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); DCHECK(request_context_getter_); download_status_ = DOWNLOAD_IN_PROGRESS; FOR_EACH_OBSERVER(Observer, observers_, OnHunspellDictionaryDownloadBegin()); fetcher_.reset(net::URLFetcher::Create(url, net::URLFetcher::GET, this)); fetcher_->SetRequestContext(request_context_getter_); fetcher_->SetLoadFlags( net::LOAD_DO_NOT_SEND_COOKIES | net::LOAD_DO_NOT_SAVE_COOKIES); fetcher_->Start(); // Attempt downloading the dictionary only once. request_context_getter_ = NULL; } // The default_dictionary_file can either come from the standard list of // hunspell dictionaries (determined in InitializeDictionaryLocation), or it // can be passed in via an extension. In either case, the file is checked for // existence so that it's not re-downloaded. // For systemwide installations on Windows, the default directory may not // have permissions for download. In that case, the alternate directory for // download is chrome::DIR_USER_DATA. SpellcheckHunspellDictionary::DictionaryFile SpellcheckHunspellDictionary::OpenDictionaryFile(const base::FilePath& path) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); DictionaryFile dictionary; #if defined(OS_WIN) // Check if the dictionary exists in the fallback location. If so, use it // rather than downloading anew. base::FilePath user_dir; PathService::Get(chrome::DIR_USER_DATA, &user_dir); base::FilePath fallback = user_dir.Append(path.BaseName()); if (!base::PathExists(path) && base::PathExists(fallback)) dictionary.path = fallback; else dictionary.path = path; #else dictionary.path = path; #endif // Read the dictionary file and scan its data to check for corruption. The // scoping closes the memory-mapped file before it is opened or deleted. bool bdict_is_valid; { base::MemoryMappedFile map; bdict_is_valid = base::PathExists(dictionary.path) && map.Initialize(dictionary.path) && hunspell::BDict::Verify(reinterpret_cast(map.data()), map.length()); } if (bdict_is_valid) { dictionary.file.Initialize(dictionary.path, base::File::FLAG_READ | base::File::FLAG_OPEN); } else { base::DeleteFile(dictionary.path, false); } return dictionary.Pass(); } // The default place where the spellcheck dictionary resides is // chrome::DIR_APP_DICTIONARIES. SpellcheckHunspellDictionary::DictionaryFile SpellcheckHunspellDictionary::InitializeDictionaryLocation( const std::string& language) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); // Initialize the BDICT path. Initialization should be in the FILE thread // because it checks if there is a "Dictionaries" directory and create it. base::FilePath dict_dir; PathService::Get(chrome::DIR_APP_DICTIONARIES, &dict_dir); base::FilePath dict_path = chrome::spellcheck_common::GetVersionedFileName(language, dict_dir); return OpenDictionaryFile(dict_path); } void SpellcheckHunspellDictionary::InitializeDictionaryLocationComplete( DictionaryFile file) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); dictionary_file_ = file.Pass(); if (!dictionary_file_.file.IsValid()) { // Notify browser tests that this dictionary is corrupted. Skip downloading // the dictionary in browser tests. // TODO(rouslan): Remove this test-only case. if (spellcheck_service_->SignalStatusEvent( SpellcheckService::BDICT_CORRUPTED)) { request_context_getter_ = NULL; } if (request_context_getter_) { // Download from the UI thread to check that |request_context_getter_| is // still valid. DownloadDictionary(GetDictionaryURL()); return; } } InformListenersOfInitialization(); } void SpellcheckHunspellDictionary::SaveDictionaryDataComplete( bool dictionary_saved) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); if (dictionary_saved) { download_status_ = DOWNLOAD_NONE; FOR_EACH_OBSERVER(Observer, observers_, OnHunspellDictionaryDownloadSuccess()); Load(); } else { InformListenersOfDownloadFailure(); InformListenersOfInitialization(); } } void SpellcheckHunspellDictionary::InformListenersOfInitialization() { FOR_EACH_OBSERVER(Observer, observers_, OnHunspellDictionaryInitialized()); } void SpellcheckHunspellDictionary::InformListenersOfDownloadFailure() { download_status_ = DOWNLOAD_FAILED; FOR_EACH_OBSERVER(Observer, observers_, OnHunspellDictionaryDownloadFailure()); }