Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(278)

Side by Side Diff: chrome/browser/spellchecker.cc

Issue 155394: [chromium-reviews] For system-level installs, if the spellcheck dictionary fails to get download... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: '' Created 11 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « chrome/browser/spellchecker.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2006-2009 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2006-2009 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "app/l10n_util.h" 5 #include "app/l10n_util.h"
6 #include "chrome/browser/spellchecker.h" 6 #include "chrome/browser/spellchecker.h"
7 #include "chrome/browser/spellchecker_common.h" 7 #include "chrome/browser/spellchecker_common.h"
8 #include "chrome/browser/spellchecker_platform_engine.h" 8 #include "chrome/browser/spellchecker_platform_engine.h"
9 #include "base/basictypes.h" 9 #include "base/basictypes.h"
10 #include "base/compiler_specific.h" 10 #include "base/compiler_specific.h"
(...skipping 59 matching lines...) Expand 10 before | Expand all | Expand 10 after
70 {"sk", "sk-SK"}, 70 {"sk", "sk-SK"},
71 {"sl", "sl-SI"}, 71 {"sl", "sl-SI"},
72 {"ca", "ca-ES"}, 72 {"ca", "ca-ES"},
73 {"lv", "lv-LV"}, 73 {"lv", "lv-LV"},
74 // {"uk", "uk-UA"}, // Not to be included in Spellchecker as per B=1277824 74 // {"uk", "uk-UA"}, // Not to be included in Spellchecker as per B=1277824
75 {"hi", "hi-IN"}, 75 {"hi", "hi-IN"},
76 {"et", "et-EE"}, 76 {"et", "et-EE"},
77 {"tr", "tr-TR"}, 77 {"tr", "tr-TR"},
78 }; 78 };
79 79
80 // Get the fallback folder (currently chrome::DIR_USER_DATA) where the
81 // dictionary is downloaded in case of system-wide installations.
82 FilePath GetFallbackDictionaryDownloadDirectory() {
83 FilePath dict_dir_userdata;
84 PathService::Get(chrome::DIR_USER_DATA, &dict_dir_userdata);
85 dict_dir_userdata = dict_dir_userdata.AppendASCII("Dictionaries");
86 return dict_dir_userdata;
87 }
88
80 } 89 }
81 90
82 void SpellChecker::SpellCheckLanguages(std::vector<std::string>* languages) { 91 void SpellChecker::SpellCheckLanguages(std::vector<std::string>* languages) {
83 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(g_supported_spellchecker_languages); 92 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(g_supported_spellchecker_languages);
84 ++i) 93 ++i)
85 languages->push_back(g_supported_spellchecker_languages[i].language); 94 languages->push_back(g_supported_spellchecker_languages[i].language);
86 } 95 }
87 96
88 // This function returns the language-region version of language name. 97 // This function returns the language-region version of language name.
89 // e.g. returns hi-IN for hi. 98 // e.g. returns hi-IN for hi.
(...skipping 146 matching lines...) Expand 10 before | Expand all | Expand 10 after
236 // which obtains the IO thread independently and invokes the task in the IO 245 // which obtains the IO thread independently and invokes the task in the IO
237 // thread if it's not NULL. After the flags are cleared, a (final) attempt is 246 // thread if it's not NULL. After the flags are cleared, a (final) attempt is
238 // made to initialize hunspell_. If it fails even then (dictionary could not 247 // made to initialize hunspell_. If it fails even then (dictionary could not
239 // download), no more attempts are made to initialize it. 248 // download), no more attempts are made to initialize it.
240 249
241 // ############################################################################ 250 // ############################################################################
242 251
243 // This object downloads the dictionary files asynchronously by first 252 // This object downloads the dictionary files asynchronously by first
244 // fetching it to memory using URL fetcher and then writing it to 253 // fetching it to memory using URL fetcher and then writing it to
245 // disk using file_util::WriteFile. 254 // disk using file_util::WriteFile.
255
246 class SpellChecker::DictionaryDownloadController 256 class SpellChecker::DictionaryDownloadController
247 : public URLFetcher::Delegate, 257 : public URLFetcher::Delegate,
248 public base::RefCountedThreadSafe<DictionaryDownloadController> { 258 public base::RefCountedThreadSafe<DictionaryDownloadController> {
249 public: 259 public:
250 DictionaryDownloadController( 260 DictionaryDownloadController(
251 Task* spellchecker_flag_set_task, 261 Task* spellchecker_flag_set_task,
252 const FilePath& dic_file_path, 262 const FilePath& dic_file_path,
253 URLRequestContext* url_request_context, 263 URLRequestContext* url_request_context,
254 MessageLoop* ui_loop) 264 MessageLoop* ui_loop)
255 : spellchecker_flag_set_task_(spellchecker_flag_set_task), 265 : spellchecker_flag_set_task_(spellchecker_flag_set_task),
(...skipping 13 matching lines...) Expand all
269 279
270 GURL url = GURL(std::string(kDownloadServerUrl) + WideToUTF8( 280 GURL url = GURL(std::string(kDownloadServerUrl) + WideToUTF8(
271 l10n_util::ToLower(file_name_.ToWStringHack()))); 281 l10n_util::ToLower(file_name_.ToWStringHack())));
272 fetcher_.reset(new URLFetcher(url, URLFetcher::GET, this)); 282 fetcher_.reset(new URLFetcher(url, URLFetcher::GET, this));
273 fetcher_->set_request_context(url_request_context_); 283 fetcher_->set_request_context(url_request_context_);
274 fetcher_->Start(); 284 fetcher_->Start();
275 } 285 }
276 286
277 private: 287 private:
278 // The file has been downloaded in memory - need to write it down to file. 288 // The file has been downloaded in memory - need to write it down to file.
279 bool SaveBufferToFile(const std::string& data) { 289 bool SaveBufferToFile(const std::string& data,
280 FilePath file_to_write = dic_zip_file_path_.Append(file_name_); 290 FilePath file_to_write) {
281 int num_bytes = data.length(); 291 int num_bytes = data.length();
282 return file_util::WriteFile(file_to_write, data.data(), num_bytes) == 292 return file_util::WriteFile(file_to_write, data.data(), num_bytes) ==
283 num_bytes; 293 num_bytes;
284 } 294 }
285 295
286 // URLFetcher::Delegate interface. 296 // URLFetcher::Delegate interface.
287 virtual void OnURLFetchComplete(const URLFetcher* source, 297 virtual void OnURLFetchComplete(const URLFetcher* source,
288 const GURL& url, 298 const GURL& url,
289 const URLRequestStatus& status, 299 const URLRequestStatus& status,
290 int response_code, 300 int response_code,
291 const ResponseCookies& cookies, 301 const ResponseCookies& cookies,
292 const std::string& data) { 302 const std::string& data) {
293 DCHECK(source); 303 DCHECK(source);
294 bool save_success = false;
295 if ((response_code / 100) == 2 || 304 if ((response_code / 100) == 2 ||
296 response_code == 401 || 305 response_code == 401 ||
297 response_code == 407) { 306 response_code == 407) {
298 save_success = SaveBufferToFile(data); 307 FilePath file_to_write = dic_zip_file_path_.Append(file_name_);
308 if (!SaveBufferToFile(data, file_to_write)) {
309 // Try saving it to user data/Dictionaries, which almost surely has
310 // write permission. If even this fails, there is nothing to be done.
311 FilePath user_data_dir = GetFallbackDictionaryDownloadDirectory();
312
313 // Create the directory if it does not exist.
314 if (!file_util::PathExists(user_data_dir))
315 file_util::CreateDirectory(user_data_dir);
316
317 file_to_write = user_data_dir.Append(file_name_);
318 SaveBufferToFile(data.data(), file_to_write);
319 }
299 } // Unsuccessful save is taken care of in SpellChecker::Initialize(). 320 } // Unsuccessful save is taken care of in SpellChecker::Initialize().
300 321
301 // Set Flag that dictionary is not downloading anymore. 322 // Set Flag that dictionary is not downloading anymore.
302 ui_loop_->PostTask(FROM_HERE, 323 ui_loop_->PostTask(FROM_HERE,
303 new UIProxyForIOTask(spellchecker_flag_set_task_)); 324 new UIProxyForIOTask(spellchecker_flag_set_task_));
304 fetcher_.reset(NULL); 325 fetcher_.reset(NULL);
305 } 326 }
306 327
307 // factory object to invokelater back to spellchecker in io thread on 328 // factory object to invokelater back to spellchecker in io thread on
308 // download completion to change appropriate flags. 329 // download completion to change appropriate flags.
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after
379 } 400 }
380 } 401 }
381 402
382 return dict_dir.AppendASCII(versioned_bdict_file_name); 403 return dict_dir.AppendASCII(versioned_bdict_file_name);
383 } 404 }
384 405
385 SpellChecker::SpellChecker(const FilePath& dict_dir, 406 SpellChecker::SpellChecker(const FilePath& dict_dir,
386 const std::string& language, 407 const std::string& language,
387 URLRequestContext* request_context, 408 URLRequestContext* request_context,
388 const FilePath& custom_dictionary_file_name) 409 const FilePath& custom_dictionary_file_name)
389 : custom_dictionary_file_name_(custom_dictionary_file_name), 410 : given_dictionary_directory_(dict_dir),
411 custom_dictionary_file_name_(custom_dictionary_file_name),
390 tried_to_init_(false), 412 tried_to_init_(false),
413 language_(language),
391 #ifndef NDEBUG 414 #ifndef NDEBUG
392 worker_loop_(NULL), 415 worker_loop_(NULL),
393 #endif 416 #endif
394 tried_to_download_(false), 417 tried_to_download_dictionary_file_(false),
395 file_loop_(NULL), 418 file_loop_(NULL),
396 url_request_context_(request_context), 419 url_request_context_(request_context),
397 dic_is_downloading_(false), 420 dic_is_downloading_(false),
398 auto_spell_correct_turned_on_(false), 421 auto_spell_correct_turned_on_(false),
399 is_using_platform_spelling_engine_(false), 422 is_using_platform_spelling_engine_(false),
400 ALLOW_THIS_IN_INITIALIZER_LIST( 423 ALLOW_THIS_IN_INITIALIZER_LIST(
401 dic_download_state_changer_factory_(this)) { 424 dic_download_state_changer_factory_(this)) {
402 if (SpellCheckerPlatform::SpellCheckerAvailable()) { 425 if (SpellCheckerPlatform::SpellCheckerAvailable()) {
403 SpellCheckerPlatform::Init(); 426 SpellCheckerPlatform::Init();
404 if (SpellCheckerPlatform::PlatformSupportsLanguage(language)) { 427 if (SpellCheckerPlatform::PlatformSupportsLanguage(language)) {
405 // If we have reached here, then we know that the current platform 428 // If we have reached here, then we know that the current platform
406 // supports the given language and we will use it instead of hunspell. 429 // supports the given language and we will use it instead of hunspell.
407 SpellCheckerPlatform::SetLanguage(language); 430 SpellCheckerPlatform::SetLanguage(language);
408 is_using_platform_spelling_engine_ = true; 431 is_using_platform_spelling_engine_ = true;
409 } 432 }
410 } 433 }
411 434
412 // Remember UI loop to later use this as a proxy to get IO loop. 435 // Remember UI loop to later use this as a proxy to get IO loop.
413 ui_loop_ = MessageLoop::current(); 436 ui_loop_ = MessageLoop::current();
414 437
415 // Get File Loop - hunspell gets initialized here. 438 // Get File Loop - hunspell gets initialized here.
416 base::Thread* file_thread = g_browser_process->file_thread(); 439 base::Thread* file_thread = g_browser_process->file_thread();
417 if (file_thread) 440 if (file_thread)
418 file_loop_ = file_thread->message_loop(); 441 file_loop_ = file_thread->message_loop();
419 442
420 // Get the path to the spellcheck file.
421 bdict_file_name_ = GetVersionedFileName(language, dict_dir);
422
423 // Get the path to the custom dictionary file. 443 // Get the path to the custom dictionary file.
424 if (custom_dictionary_file_name_.empty()) { 444 if (custom_dictionary_file_name_.empty()) {
425 FilePath personal_file_directory; 445 FilePath personal_file_directory;
426 PathService::Get(chrome::DIR_USER_DATA, &personal_file_directory); 446 PathService::Get(chrome::DIR_USER_DATA, &personal_file_directory);
427 custom_dictionary_file_name_ = 447 custom_dictionary_file_name_ =
428 personal_file_directory.Append(chrome::kCustomDictionaryFileName); 448 personal_file_directory.Append(chrome::kCustomDictionaryFileName);
429 } 449 }
430 450
431 // Use this dictionary language as the default one of the 451 // Use this dictionary language as the default one of the
432 // SpellcheckCharAttribute object. 452 // SpellcheckCharAttribute object.
433 character_attributes_.SetDefaultLanguage(language); 453 character_attributes_.SetDefaultLanguage(language);
434 } 454 }
435 455
436 SpellChecker::~SpellChecker() { 456 SpellChecker::~SpellChecker() {
437 #ifndef NDEBUG 457 #ifndef NDEBUG
438 // This must be deleted on the I/O thread (see the header). This is the same 458 // This must be deleted on the I/O thread (see the header). This is the same
439 // thread thatSpellCheckWord is called on, so we verify that they were all the 459 // thread thatSpellCheckWord is called on, so we verify that they were all the
440 // same thread. 460 // same thread.
441 if (worker_loop_) 461 if (worker_loop_)
442 DCHECK(MessageLoop::current() == worker_loop_); 462 DCHECK(MessageLoop::current() == worker_loop_);
443 #endif 463 #endif
444 } 464 }
445 465
466 void SpellChecker::StartDictionaryDownloadInFileThread(
467 const FilePath& file_name) {
468 Task* dic_task = dic_download_state_changer_factory_.NewRunnableMethod(
469 &SpellChecker::set_file_is_downloading, false);
470 ddc_dic_ = new DictionaryDownloadController(dic_task, file_name,
471 url_request_context_, ui_loop_);
472 set_file_is_downloading(true);
473 file_loop_->PostTask(FROM_HERE, NewRunnableMethod(ddc_dic_.get(),
474 &DictionaryDownloadController::StartDownload));
475 }
476
446 // Initialize SpellChecker. In this method, if the dicitonary is not present 477 // Initialize SpellChecker. In this method, if the dicitonary is not present
447 // in the local disk, it is fetched asynchronously. 478 // in the local disk, it is fetched asynchronously.
448 // TODO(sidchat): After dictionary is downloaded, initialize hunspell in 479 // TODO(sidchat): After dictionary is downloaded, initialize hunspell in
449 // file loop - this is currently being done in the io loop. 480 // file loop - this is currently being done in the io loop.
450 // Bug: http://b/issue?id=1123096 481 // Bug: http://b/issue?id=1123096
451 bool SpellChecker::Initialize() { 482 bool SpellChecker::Initialize() {
452 // Return false if the dictionary files are downloading. 483 // Return false if the dictionary files are downloading.
453 if (dic_is_downloading_) 484 if (dic_is_downloading_)
454 return false; 485 return false;
455 486
456 // Return false if tried to init and failed - don't try multiple times in 487 // Return false if tried to init and failed - don't try multiple times in
457 // this session. 488 // this session.
458 if (tried_to_init_) 489 if (tried_to_init_)
459 return hunspell_.get() != NULL; 490 return hunspell_.get() != NULL;
460 491
461 StatsScope<StatsCounterTimer> timer(chrome::Counters::spellcheck_init()); 492 StatsScope<StatsCounterTimer> timer(chrome::Counters::spellcheck_init());
462 493
463 bool dic_exists = file_util::PathExists(bdict_file_name_); 494 // The default place whether the spellcheck dictionary can reside is
464 if (!dic_exists) { 495 // chrome::DIR_APP_DICTIONARIES. However, for systemwide installations,
465 if (file_loop_ && !tried_to_download_ && url_request_context_) { 496 // this directory may not have permissions for download. In that case, the
466 Task* dic_task = dic_download_state_changer_factory_.NewRunnableMethod( 497 // alternate directory for download is chrome::DIR_USER_DATA. We have to check
467 &SpellChecker::set_file_is_downloading, false); 498 // for the spellcheck dictionaries in both the directories. If not found in
468 ddc_dic_ = new DictionaryDownloadController(dic_task, bdict_file_name_, 499 // either one, it has to be downloaded in either of the two.
469 url_request_context_, ui_loop_); 500 // TODO(sidchat): Some sort of UI to warn users that spellchecker is not
470 set_file_is_downloading(true); 501 // working at all (due to failed dictionary download)?
471 file_loop_->PostTask(FROM_HERE, NewRunnableMethod(ddc_dic_.get(), 502
472 &DictionaryDownloadController::StartDownload)); 503 // File name for downloading in DIR_APP_DICTIONARIES.
504 FilePath dictionary_file_name_app = GetVersionedFileName(language_,
505 given_dictionary_directory_);
506
507 // Filename for downloading in the fallback dictionary download directory,
508 // DIR_USER_DATA.
509 FilePath dict_dir_userdata = GetFallbackDictionaryDownloadDirectory();
510 FilePath dictionary_file_name_usr = GetVersionedFileName(language_,
511 dict_dir_userdata);
512
513 // Check in both the directories to see whether the spellcheck dictionary
514 // already resides in one of these.
515 FilePath bdic_file_name;
516 if (file_util::PathExists(dictionary_file_name_app)) {
517 bdic_file_name = dictionary_file_name_app;
518 } else if (file_util::PathExists(dictionary_file_name_usr)) {
519 bdic_file_name = dictionary_file_name_usr;
520 } else {
521 // Download the dictionary file.
522 if (file_loop_ && url_request_context_) {
523 if (!tried_to_download_dictionary_file_) {
524 StartDictionaryDownloadInFileThread(dictionary_file_name_app);
525 tried_to_download_dictionary_file_ = true;
526 return false;
527 } else { // There is no dictionary even after trying to download it.
528 // Stop trying to download the dictionary in this session.
529 tried_to_init_ = true;
530 return false;
531 }
473 } 532 }
474 } 533 }
475 534
476 if (!dic_exists && !tried_to_download_) { 535 // Control has come so far - the BDIC dictionary file probably exists. Now try
477 tried_to_download_ = true; 536 // to initialize hunspell using the available bdic dictionary file.
478 return false;
479 }
480
481 // Control has come so far - both files probably exist.
482 TimeTicks begin_time = TimeTicks::Now(); 537 TimeTicks begin_time = TimeTicks::Now();
483 bdict_file_.reset(new file_util::MemoryMappedFile()); 538 bdict_file_.reset(new file_util::MemoryMappedFile());
484 if (bdict_file_->Initialize(bdict_file_name_)) { 539 if (bdict_file_->Initialize(bdic_file_name)) {
485 hunspell_.reset(new Hunspell(bdict_file_->data(), bdict_file_->length())); 540 hunspell_.reset(new Hunspell(bdict_file_->data(), bdict_file_->length()));
486 AddCustomWordsToHunspell(); 541 AddCustomWordsToHunspell();
487 } 542 }
488 DHISTOGRAM_TIMES("Spellcheck.InitTime", TimeTicks::Now() - begin_time); 543 DHISTOGRAM_TIMES("Spellcheck.InitTime", TimeTicks::Now() - begin_time);
489 544
490 tried_to_init_ = true; 545 tried_to_init_ = true;
491 return false; 546 return false;
492 } 547 }
493 548
494 void SpellChecker::GetAutoCorrectionWord(const std::wstring& word, 549 void SpellChecker::GetAutoCorrectionWord(const std::wstring& word,
(...skipping 232 matching lines...) Expand 10 before | Expand all | Expand 10 after
727 782
728 // Populate the vector of WideStrings. 783 // Populate the vector of WideStrings.
729 for (int i = 0; i < number_of_suggestions; i++) { 784 for (int i = 0; i < number_of_suggestions; i++) {
730 if (i < kMaxSuggestions) 785 if (i < kMaxSuggestions)
731 optional_suggestions->push_back(UTF8ToWide(suggestions[i])); 786 optional_suggestions->push_back(UTF8ToWide(suggestions[i]));
732 free(suggestions[i]); 787 free(suggestions[i]);
733 } 788 }
734 if (suggestions != NULL) 789 if (suggestions != NULL)
735 free(suggestions); 790 free(suggestions);
736 } 791 }
OLDNEW
« no previous file with comments | « chrome/browser/spellchecker.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698