autofill_profile_data_type_controller.cc revision f8ee788a64d60abd8f2d742a5fdedde054ecd910
1// Copyright (c) 2012 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "chrome/browser/sync/glue/autofill_profile_data_type_controller.h"
6
7#include "base/bind.h"
8#include "base/metrics/histogram.h"
9#include "chrome/browser/autofill/personal_data_manager_factory.h"
10#include "chrome/browser/profiles/profile.h"
11#include "chrome/browser/sync/glue/chrome_report_unrecoverable_error.h"
12#include "chrome/browser/sync/profile_sync_components_factory.h"
13#include "chrome/browser/webdata/web_data_service_factory.h"
14#include "components/autofill/core/browser/personal_data_manager.h"
15#include "components/autofill/core/browser/webdata/autofill_webdata_service.h"
16#include "content/public/browser/browser_thread.h"
17#include "sync/api/sync_error.h"
18#include "sync/api/syncable_service.h"
19
20using autofill::AutofillWebDataService;
21using content::BrowserThread;
22
23namespace browser_sync {
24
25AutofillProfileDataTypeController::AutofillProfileDataTypeController(
26    ProfileSyncComponentsFactory* profile_sync_factory,
27    Profile* profile,
28    const DisableTypeCallback& disable_callback)
29    : NonUIDataTypeController(
30          BrowserThread::GetMessageLoopProxyForThread(BrowserThread::UI),
31          base::Bind(&ChromeReportUnrecoverableError),
32          disable_callback,
33          profile_sync_factory),
34      profile_(profile),
35      personal_data_(NULL),
36      callback_registered_(false) {}
37
38syncer::ModelType AutofillProfileDataTypeController::type() const {
39  return syncer::AUTOFILL_PROFILE;
40}
41
42syncer::ModelSafeGroup
43    AutofillProfileDataTypeController::model_safe_group() const {
44  return syncer::GROUP_DB;
45}
46
47void AutofillProfileDataTypeController::WebDatabaseLoaded() {
48  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
49  OnModelLoaded();
50}
51
52void AutofillProfileDataTypeController::OnPersonalDataChanged() {
53  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
54  DCHECK_EQ(state(), MODEL_STARTING);
55
56  personal_data_->RemoveObserver(this);
57  autofill::AutofillWebDataService* web_data_service =
58      WebDataServiceFactory::GetAutofillWebDataForProfile(
59          profile_, Profile::EXPLICIT_ACCESS).get();
60
61  if (!web_data_service)
62    return;
63
64  if (web_data_service->IsDatabaseLoaded()) {
65    OnModelLoaded();
66  } else  if (!callback_registered_) {
67     web_data_service->RegisterDBLoadedCallback(base::Bind(
68        &AutofillProfileDataTypeController::WebDatabaseLoaded, this));
69     callback_registered_ = true;
70  }
71}
72
73AutofillProfileDataTypeController::~AutofillProfileDataTypeController() {}
74
75bool AutofillProfileDataTypeController::PostTaskOnBackendThread(
76    const tracked_objects::Location& from_here,
77    const base::Closure& task) {
78  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
79  return BrowserThread::PostTask(BrowserThread::DB, from_here, task);
80}
81
82bool AutofillProfileDataTypeController::StartModels() {
83  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
84  DCHECK_EQ(state(), MODEL_STARTING);
85  // Waiting for the personal data is subtle:  we do this as the PDM resets
86  // its cache of unique IDs once it gets loaded. If we were to proceed with
87  // association, the local ids in the mappings would wind up colliding.
88  personal_data_ =
89      autofill::PersonalDataManagerFactory::GetForProfile(profile_);
90  if (!personal_data_->IsDataLoaded()) {
91    personal_data_->AddObserver(this);
92    return false;
93  }
94
95  autofill::AutofillWebDataService* web_data_service =
96      WebDataServiceFactory::GetAutofillWebDataForProfile(
97          profile_, Profile::EXPLICIT_ACCESS).get();
98
99  if (!web_data_service)
100    return false;
101
102  if (web_data_service->IsDatabaseLoaded())
103    return true;
104
105  if (!callback_registered_) {
106     web_data_service->RegisterDBLoadedCallback(base::Bind(
107        &AutofillProfileDataTypeController::WebDatabaseLoaded, this));
108     callback_registered_ = true;
109  }
110
111  return false;
112}
113
114void AutofillProfileDataTypeController::StopModels() {
115  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
116  personal_data_->RemoveObserver(this);
117}
118
119}  // namespace browser_sync
120