blob: 29134d59cc0c872125701c15b6614feb7aa85110 [file] [log] [blame]
// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/bookmarks/browser/model_loader.h"
#include <optional>
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/functional/bind.h"
#include "base/json/json_file_value_serializer.h"
#include "base/json/json_reader.h"
#include "base/numerics/clamped_math.h"
#include "base/synchronization/waitable_event.h"
#include "base/task/task_traits.h"
#include "base/task/thread_pool.h"
#include "components/bookmarks/browser/bookmark_codec.h"
#include "components/bookmarks/browser/bookmark_load_details.h"
#include "components/bookmarks/browser/titled_url_index.h"
#include "components/bookmarks/browser/url_index.h"
#include "components/bookmarks/common/bookmark_metrics.h"
#include "components/bookmarks/common/url_load_stats.h"
#include "components/bookmarks/common/user_folder_load_stats.h"
namespace bookmarks {
namespace {
// Loads and deserializes a JSON file determined by `file_path` and returns it
// in the form of a dictionary, or nullopt if something fails.
std::optional<base::Value::Dict> LoadFileToDict(
const base::FilePath& file_path) {
// Titles may end up containing invalid utf and we shouldn't throw away
// all bookmarks if some titles have invalid utf.
JSONFileValueDeserializer deserializer(file_path,
base::JSON_REPLACE_INVALID_CHARACTERS);
std::unique_ptr<base::Value> root = deserializer.Deserialize(
/*error_code=*/nullptr, /*error_message=*/nullptr);
if (!root || !root->is_dict()) {
// The bookmark file exists but was not deserialized properly.
return std::nullopt;
}
return std::make_optional(std::move(*root).TakeDict());
}
std::unique_ptr<BookmarkLoadDetails> LoadBookmarks(
const base::FilePath& local_or_syncable_file_path,
const base::FilePath& account_file_path) {
auto details = std::make_unique<BookmarkLoadDetails>();
std::set<int64_t> ids_assigned_to_account_nodes;
// Decode account bookmarks (if any). Doing this before decoding
// local-or-syncable ones is interesting because, in case there are ID
// collisions, it will lead to ID reassignments on the local-or-syncable part,
// which is usually harmless. Doing the opposite would imply that account
// bookmarks need to be redownloaded from the server (because ID reassignment
// leads to invalidating sync metadata). This is particularly interesting on
// iOS, in case the files were written by two independent BookmarkModel
// instances (and hence the two files are prone to ID collisions) and later
// loaded into a single BookmarkModel instance.
if (!account_file_path.empty()) {
std::string sync_metadata_str;
int64_t max_node_id = 0;
std::unique_ptr<BookmarkPermanentNode> account_bb_node =
BookmarkPermanentNode::CreateBookmarkBar(0);
std::unique_ptr<BookmarkPermanentNode> account_other_folder_node =
BookmarkPermanentNode::CreateOtherBookmarks(0);
std::unique_ptr<BookmarkPermanentNode> account_mobile_folder_node =
BookmarkPermanentNode::CreateMobileBookmarks(0);
std::optional<base::Value::Dict> root_dict =
LoadFileToDict(account_file_path);
BookmarkCodec codec;
if (root_dict.has_value() &&
codec.Decode(*root_dict, /*already_assigned_ids=*/{},
account_bb_node.get(), account_other_folder_node.get(),
account_mobile_folder_node.get(), &max_node_id,
&sync_metadata_str)) {
ids_assigned_to_account_nodes = codec.release_assigned_ids();
// A successful decoding must have set proper IDs.
CHECK_NE(0, account_bb_node->id());
CHECK_NE(0, account_other_folder_node->id());
CHECK_NE(0, account_mobile_folder_node->id());
details->AddAccountPermanentNodes(std::move(account_bb_node),
std::move(account_other_folder_node),
std::move(account_mobile_folder_node));
details->set_account_sync_metadata_str(std::move(sync_metadata_str));
details->set_max_id(std::max(max_node_id, details->max_id()));
details->set_ids_reassigned(details->ids_reassigned() ||
codec.ids_reassigned());
details->set_required_recovery(details->required_recovery() ||
codec.required_recovery());
// Record metrics that indicate whether or not IDs were reassigned for
// account bookmarks.
metrics::RecordIdsReassignedOnProfileLoad(
metrics::StorageFileForUma::kAccount, codec.ids_reassigned());
} else {
// In the failure case, it is still possible that sync metadata was
// decoded, which includes legit scenarios like sync metadata indicating
// that there were too many bookmarks in sync, server-side.
details->set_account_sync_metadata_str(std::move(sync_metadata_str));
}
}
// Decode local-or-syncable bookmarks.
{
std::string sync_metadata_str;
int64_t max_node_id = 0;
std::optional<base::Value::Dict> root_dict =
LoadFileToDict(local_or_syncable_file_path);
BookmarkCodec codec;
if (root_dict.has_value() &&
codec.Decode(*root_dict, std::move(ids_assigned_to_account_nodes),
details->bb_node(), details->other_folder_node(),
details->mobile_folder_node(), &max_node_id,
&sync_metadata_str)) {
details->set_local_or_syncable_sync_metadata_str(
std::move(sync_metadata_str));
details->set_max_id(std::max(max_node_id, details->max_id()));
details->set_ids_reassigned(details->ids_reassigned() ||
codec.ids_reassigned());
details->set_required_recovery(details->required_recovery() ||
codec.required_recovery());
details->set_local_or_syncable_reassigned_ids_per_old_id(
codec.release_reassigned_ids_per_old_id());
// Record metrics that indicate whether or not IDs were reassigned for
// local-or-syncable bookmarks.
metrics::RecordIdsReassignedOnProfileLoad(
metrics::StorageFileForUma::kLocalOrSyncable, codec.ids_reassigned());
}
}
return details;
}
void LoadManagedNode(LoadManagedNodeCallback load_managed_node_callback,
BookmarkLoadDetails& details) {
if (!load_managed_node_callback) {
return;
}
int64_t max_node_id = details.max_id();
std::unique_ptr<BookmarkPermanentNode> managed_node =
std::move(load_managed_node_callback).Run(&max_node_id);
if (managed_node) {
details.AddManagedNode(std::move(managed_node));
details.set_max_id(std::max(max_node_id, details.max_id()));
}
}
uint64_t GetFileSizeOrZero(const base::FilePath& file_path) {
return base::GetFileSize(file_path).value_or(0);
}
void RecordLoadMetrics(BookmarkLoadDetails* details,
const base::FilePath& local_or_syncable_file_path,
const base::FilePath& account_file_path) {
UrlLoadStats url_stats = details->url_index()->ComputeStats();
metrics::RecordUrlLoadStatsOnProfileLoad(url_stats);
UserFolderLoadStats user_folder_stats = details->ComputeUserFolderStats();
metrics::RecordUserFolderLoadStatsOnProfileLoad(user_folder_stats);
const uint64_t local_or_syncable_file_size =
GetFileSizeOrZero(local_or_syncable_file_path);
const uint64_t account_file_size =
account_file_path.empty() ? 0U : GetFileSizeOrZero(account_file_path);
if (local_or_syncable_file_size != 0) {
metrics::RecordFileSizeAtStartup(local_or_syncable_file_size);
}
if (account_file_size != 0) {
metrics::RecordFileSizeAtStartup(account_file_size);
}
const uint64_t sum_file_size =
local_or_syncable_file_size + account_file_size;
if (sum_file_size > 0) {
metrics::RecordAverageNodeSizeAtStartup(
url_stats.total_url_bookmark_count == 0
? 0
: sum_file_size / url_stats.total_url_bookmark_count);
}
}
} // namespace
// static
scoped_refptr<ModelLoader> ModelLoader::Create(
const base::FilePath& local_or_syncable_file_path,
const base::FilePath& account_file_path,
LoadManagedNodeCallback load_managed_node_callback,
LoadCallback callback) {
CHECK(!local_or_syncable_file_path.empty());
// Note: base::MakeRefCounted is not available here, as ModelLoader's
// constructor is private.
auto model_loader = base::WrapRefCounted(new ModelLoader());
model_loader->backend_task_runner_ =
base::ThreadPool::CreateSequencedTaskRunner(
{base::MayBlock(), base::TaskPriority::USER_VISIBLE,
base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN});
model_loader->backend_task_runner_->PostTaskAndReplyWithResult(
FROM_HERE,
base::BindOnce(&ModelLoader::DoLoadOnBackgroundThread, model_loader,
local_or_syncable_file_path, account_file_path,
std::move(load_managed_node_callback)),
std::move(callback));
return model_loader;
}
void ModelLoader::BlockTillLoaded() {
loaded_signal_.Wait();
}
ModelLoader::ModelLoader()
: loaded_signal_(base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED) {}
ModelLoader::~ModelLoader() = default;
std::unique_ptr<BookmarkLoadDetails> ModelLoader::DoLoadOnBackgroundThread(
const base::FilePath& local_or_syncable_file_path,
const base::FilePath& account_file_path,
LoadManagedNodeCallback load_managed_node_callback) {
std::unique_ptr<BookmarkLoadDetails> details =
LoadBookmarks(local_or_syncable_file_path, account_file_path);
CHECK(details);
details->PopulateNodeIdsForLocalOrSyncablePermanentNodes();
LoadManagedNode(std::move(load_managed_node_callback), *details);
// Building the indices can take a while so it's done on the background
// thread.
details->CreateIndices();
RecordLoadMetrics(details.get(), local_or_syncable_file_path,
account_file_path);
history_bookmark_model_ = details->url_index();
loaded_signal_.Signal();
return details;
}
// static
scoped_refptr<ModelLoader> ModelLoader::CreateForTest(
LoadManagedNodeCallback load_managed_node_callback,
BookmarkLoadDetails* details) {
CHECK(details);
details->PopulateNodeIdsForLocalOrSyncablePermanentNodes();
LoadManagedNode(std::move(load_managed_node_callback), *details);
details->CreateIndices();
// Note: base::MakeRefCounted is not available here, as ModelLoader's
// constructor is private.
auto model_loader = base::WrapRefCounted(new ModelLoader());
model_loader->history_bookmark_model_ = details->url_index();
model_loader->loaded_signal_.Signal();
return model_loader;
}
} // namespace bookmarks