// Copyright 2014 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 "components/enhanced_bookmarks/metadata_accessor.h"
#include <iomanip>
#include "base/base64.h"
#include "base/rand_util.h"
#include "components/bookmarks/browser/bookmark_model.h"
#include "components/enhanced_bookmarks/proto/metadata.pb.h"
#include "ui/base/models/tree_node_iterator.h"
using namespace image::collections;
namespace {
// Helper method for working with bookmark metainfo.
std::string DataForMetaInfoField(const BookmarkNode* node,
const std::string& field) {
const BookmarkNode::MetaInfoMap* map = node->GetMetaInfoMap();
if (!map)
return "";
BookmarkNode::MetaInfoMap::const_iterator it = map->find(field);
if (it == map->end())
return "";
std::string decoded;
bool result = base::Base64Decode((*it).second, &decoded);
if (!result)
return "";
return decoded;
}
// Sets a new remote id on a bookmark.
std::string SetRemoteIdOnBookmark(BookmarkModel* bookmark_model,
const BookmarkNode* node) {
// Generate 16 digit hex string random id.
std::stringstream random_id;
random_id << std::hex << std::setfill('0') << std::setw(16);
random_id << base::RandUint64() << base::RandUint64();
std::string random_id_str = random_id.str();
bookmark_model->SetNodeMetaInfo(
node, enhanced_bookmarks::kIdDataKey, random_id_str);
return random_id_str;
}
// Helper method for working with ImageData_ImageInfo.
bool PopulateImageData(const ImageData_ImageInfo& info,
GURL* out_url,
int* width,
int* height) {
if (!info.has_url() || !info.has_width() || !info.has_height())
return false;
GURL url(info.url());
if (!url.is_valid())
return false;
*out_url = url;
*width = info.width();
*height = info.height();
return true;
}
} // namespace
namespace enhanced_bookmarks {
const char* kPageDataKey = "stars.pageData";
const char* kImageDataKey = "stars.imageData";
const char* kIdDataKey = "stars.id";
const char* kNoteKey = "stars.note";
std::string RemoteIdFromBookmark(BookmarkModel* bookmark_model,
const BookmarkNode* node) {
const BookmarkNode::MetaInfoMap* map = node->GetMetaInfoMap();
if (!map)
return SetRemoteIdOnBookmark(bookmark_model, node);
BookmarkNode::MetaInfoMap::const_iterator it = map->find(kIdDataKey);
if (it == map->end())
return SetRemoteIdOnBookmark(bookmark_model, node);
DCHECK(it->second.length());
return it->second;
}
void SetDescriptionForBookmark(BookmarkModel* bookmark_model,
const BookmarkNode* node,
const std::string& description) {
bookmark_model->SetNodeMetaInfo(node, kNoteKey, description);
}
std::string DescriptionFromBookmark(const BookmarkNode* node) {
const BookmarkNode::MetaInfoMap* map = node->GetMetaInfoMap();
if (!map)
return "";
// First, look for a custom note set by the user.
BookmarkNode::MetaInfoMap::const_iterator it = map->find(kNoteKey);
if (it != map->end() && it->second != "")
return it->second;
// If none are present, return the snippet.
return SnippetFromBookmark(node);
}
bool SetOriginalImageForBookmark(BookmarkModel* bookmark_model,
const BookmarkNode* node,
const GURL& url,
int width,
int height) {
DCHECK(url.is_valid());
std::string decoded(DataForMetaInfoField(node, kImageDataKey));
ImageData data;
// Try to populate the imageData with the existing data.
if (decoded != "") {
// If the parsing fails, something is wrong. Immediately fail.
bool result = data.ParseFromString(decoded);
if (!result)
return false;
}
scoped_ptr<ImageData_ImageInfo> info(new ImageData_ImageInfo);
info->set_url(url.spec());
info->set_width(width);
info->set_height(height);
data.set_allocated_original_info(info.release());
std::string output;
bool result = data.SerializePartialToString(&output);
if (!result)
return false;
std::string encoded;
base::Base64Encode(output, &encoded);
bookmark_model->SetNodeMetaInfo(node, kImageDataKey, encoded);
// Ensure that the bookmark has a stars.id, to trigger the server processing.
RemoteIdFromBookmark(bookmark_model, node);
return true;
}
bool OriginalImageFromBookmark(const BookmarkNode* node,
GURL* url,
int* width,
int* height) {
std::string decoded(DataForMetaInfoField(node, kImageDataKey));
if (decoded == "")
return false;
ImageData data;
bool result = data.ParseFromString(decoded);
if (!result)
return false;
if (!data.has_original_info())
return false;
return PopulateImageData(data.original_info(), url, width, height);
}
bool ThumbnailImageFromBookmark(const BookmarkNode* node,
GURL* url,
int* width,
int* height) {
std::string decoded(DataForMetaInfoField(node, kImageDataKey));
if (decoded == "")
return false;
ImageData data;
bool result = data.ParseFromString(decoded);
if (!result)
return false;
if (!data.has_thumbnail_info())
return false;
return PopulateImageData(data.thumbnail_info(), url, width, height);
}
std::string SnippetFromBookmark(const BookmarkNode* node) {
std::string decoded(DataForMetaInfoField(node, kPageDataKey));
if (decoded == "")
return decoded;
PageData data;
bool result = data.ParseFromString(decoded);
if (!result)
return "";
return data.snippet();
}
bool SetAllImagesForBookmark(BookmarkModel* bookmark_model,
const BookmarkNode* node,
const GURL& image_url,
int image_width,
int image_height,
const GURL& thumbnail_url,
int thumbnail_width,
int thumbnail_height) {
DCHECK(image_url.is_valid() || image_url.is_empty());
DCHECK(thumbnail_url.is_valid() || thumbnail_url.is_empty());
std::string decoded(DataForMetaInfoField(node, kImageDataKey));
ImageData data;
// Try to populate the imageData with the existing data.
if (decoded != "") {
// If the parsing fails, something is wrong. Immediately fail.
bool result = data.ParseFromString(decoded);
if (!result)
return false;
}
if (image_url.is_empty()) {
data.release_original_info();
} else {
// Regardless of whether an image info exists, we make a new one.
// Intentially make a raw pointer.
ImageData_ImageInfo* info = new ImageData_ImageInfo;
info->set_url(image_url.spec());
info->set_width(image_width);
info->set_height(image_height);
// This method consumes the raw pointer.
data.set_allocated_original_info(info);
}
if (thumbnail_url.is_empty()) {
data.release_thumbnail_info();
} else {
// Regardless of whether an image info exists, we make a new one.
// Intentially make a raw pointer.
ImageData_ImageInfo* info = new ImageData_ImageInfo;
info->set_url(thumbnail_url.spec());
info->set_width(thumbnail_width);
info->set_height(thumbnail_height);
// This method consumes the raw pointer.
data.set_allocated_thumbnail_info(info);
}
std::string output;
bool result = data.SerializePartialToString(&output);
if (!result)
return false;
std::string encoded;
base::Base64Encode(output, &encoded);
bookmark_model->SetNodeMetaInfo(node, kImageDataKey, encoded);
return true;
}
} // namespace enhanced_bookmarks