| // Copyright 2014 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/bookmark_node_data.h" |
| |
| #include <algorithm> |
| #include <string> |
| |
| #include "base/containers/span.h" |
| #include "base/logging.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/numerics/safe_conversions.h" |
| #include "base/pickle.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "build/build_config.h" |
| #include "components/bookmarks/browser/bookmark_utils.h" |
| #include "ui/base/clipboard/scoped_clipboard_writer.h" |
| |
| namespace bookmarks { |
| |
| #if !BUILDFLAG(IS_APPLE) |
| namespace { |
| constexpr size_t kMaxVectorPreallocateSize = 10000; |
| } // namespace |
| |
| const char BookmarkNodeData::kClipboardFormatString[] = |
| "chromium/x-bookmark-entries"; |
| #endif |
| |
| BookmarkNodeData::Element::Element() : is_url(false), id_(0) { |
| } |
| |
| BookmarkNodeData::Element::Element(const BookmarkNode* node) |
| : is_url(node->is_url()), |
| url(node->url()), |
| title(node->GetTitle()), |
| date_added(node->date_added()), |
| date_folder_modified(node->date_folder_modified()), |
| id_(node->id()) { |
| if (node->GetMetaInfoMap()) |
| meta_info_map = *node->GetMetaInfoMap(); |
| for (const auto& child : node->children()) |
| children.push_back(Element(child.get())); |
| } |
| |
| BookmarkNodeData::Element::Element(const Element& other) = default; |
| |
| BookmarkNodeData::Element::~Element() { |
| } |
| |
| #if !BUILDFLAG(IS_APPLE) |
| void BookmarkNodeData::Element::WriteToPickle(base::Pickle* pickle) const { |
| pickle->WriteBool(is_url); |
| pickle->WriteString(url.spec()); |
| pickle->WriteString16(title); |
| pickle->WriteInt64(id_); |
| pickle->WriteUInt32(static_cast<uint32_t>(meta_info_map.size())); |
| for (auto it = meta_info_map.begin(); it != meta_info_map.end(); ++it) { |
| pickle->WriteString(it->first); |
| pickle->WriteString(it->second); |
| } |
| if (!is_url) { |
| pickle->WriteUInt32(static_cast<uint32_t>(children.size())); |
| for (auto i = children.begin(); i != children.end(); ++i) { |
| i->WriteToPickle(pickle); |
| } |
| } |
| } |
| |
| bool BookmarkNodeData::Element::ReadFromPickle(base::PickleIterator* iterator) { |
| std::string url_spec; |
| if (!iterator->ReadBool(&is_url) || |
| !iterator->ReadString(&url_spec) || |
| !iterator->ReadString16(&title) || |
| !iterator->ReadInt64(&id_)) { |
| return false; |
| } |
| url = GURL(url_spec); |
| date_added = base::Time(); |
| date_folder_modified = base::Time(); |
| meta_info_map.clear(); |
| uint32_t meta_field_count; |
| if (!iterator->ReadUInt32(&meta_field_count)) |
| return false; |
| for (size_t i = 0; i < meta_field_count; ++i) { |
| std::string key; |
| std::string value; |
| if (!iterator->ReadString(&key) || |
| !iterator->ReadString(&value)) { |
| return false; |
| } |
| meta_info_map[key] = value; |
| } |
| children.clear(); |
| if (!is_url) { |
| uint32_t children_count_tmp; |
| if (!iterator->ReadUInt32(&children_count_tmp)) |
| return false; |
| if (!base::IsValueInRangeForNumericType<size_t>(children_count_tmp)) { |
| LOG(WARNING) << "children_count failed bounds check"; |
| return false; |
| } |
| // Note: do not preallocate the children vector. A pickle could be |
| // constructed to contain N nested Elements. By continually recursing on |
| // this ReadFromPickle function, the fast-fail logic is subverted. Each |
| // child would claim it contains M more children. The first (and only) child |
| // provided would claim M more children. We would allocate N * M Elements |
| // along the way, while only receiving N Elements. |
| const size_t children_count = |
| base::checked_cast<size_t>(children_count_tmp); |
| for (size_t i = 0; i < children_count; ++i) { |
| children.emplace_back(); |
| if (!children.back().ReadFromPickle(iterator)) |
| return false; |
| } |
| } |
| return true; |
| } |
| #endif |
| |
| // BookmarkNodeData ----------------------------------------------------------- |
| |
| BookmarkNodeData::BookmarkNodeData() { |
| } |
| |
| BookmarkNodeData::BookmarkNodeData(const BookmarkNodeData& other) = default; |
| |
| BookmarkNodeData::BookmarkNodeData(const BookmarkNode* node) { |
| elements.push_back(Element(node)); |
| } |
| |
| BookmarkNodeData::BookmarkNodeData( |
| const std::vector<raw_ptr<const BookmarkNode, VectorExperimental>>& nodes) { |
| ReadFromVector(nodes); |
| } |
| |
| BookmarkNodeData::~BookmarkNodeData() { |
| } |
| |
| #if !BUILDFLAG(IS_APPLE) |
| // static |
| bool BookmarkNodeData::ClipboardContainsBookmarks() { |
| ui::DataTransferEndpoint data_dst = ui::DataTransferEndpoint( |
| ui::EndpointType::kDefault, {.notify_if_restricted = false}); |
| return ui::Clipboard::GetForCurrentThread()->IsFormatAvailable( |
| ui::ClipboardFormatType::CustomPlatformType(kClipboardFormatString), |
| ui::ClipboardBuffer::kCopyPaste, &data_dst); |
| } |
| #endif |
| |
| bool BookmarkNodeData::ReadFromVector( |
| const std::vector<raw_ptr<const BookmarkNode, VectorExperimental>>& nodes) { |
| Clear(); |
| |
| if (nodes.empty()) |
| return false; |
| |
| for (size_t i = 0; i < nodes.size(); ++i) |
| elements.push_back(Element(nodes[i])); |
| |
| return true; |
| } |
| |
| bool BookmarkNodeData::ReadFromTuple(const GURL& url, |
| const std::u16string& title) { |
| Clear(); |
| |
| if (!url.is_valid()) |
| return false; |
| |
| Element element; |
| element.title = title; |
| element.url = url; |
| element.is_url = true; |
| |
| elements.push_back(element); |
| |
| return true; |
| } |
| |
| #if !BUILDFLAG(IS_APPLE) |
| void BookmarkNodeData::WriteToClipboard(bool is_off_the_record) { |
| ui::ScopedClipboardWriter scw(ui::ClipboardBuffer::kCopyPaste); |
| |
| if (is_off_the_record) { |
| // Data is copied from an incognito window, so mark it as off the record. |
| scw.MarkAsOffTheRecord(); |
| } |
| |
| #if BUILDFLAG(IS_WIN) |
| const std::u16string kEOL(u"\r\n"); |
| #else |
| const std::u16string kEOL = u"\n"; |
| #endif |
| |
| // If there is only one element and it is a URL, write the URL to the |
| // clipboard. |
| if (has_single_url()) { |
| const std::u16string& title = elements[0].title; |
| const std::string url = elements[0].url.spec(); |
| |
| scw.WriteBookmark(title, url); |
| scw.WriteHyperlink(title, url); |
| scw.WriteText(base::UTF8ToUTF16(url)); |
| } else { |
| // We have either more than one URL, a folder, or a combination of URLs |
| // and folders. |
| std::u16string text; |
| for (size_t i = 0; i < size(); i++) { |
| text += i == 0 ? u"" : kEOL; |
| if (!elements[i].is_url) { |
| // Then it's a folder. Only copy the name of the folder. |
| const std::u16string title = elements[i].title; |
| text += title; |
| } else { |
| const std::u16string url = base::UTF8ToUTF16(elements[i].url.spec()); |
| text += url; |
| } |
| } |
| scw.WriteText(text); |
| } |
| |
| base::Pickle pickle; |
| WriteToPickle(base::FilePath(), &pickle); |
| scw.WritePickledData(pickle, ui::ClipboardFormatType::CustomPlatformType( |
| kClipboardFormatString)); |
| } |
| |
| bool BookmarkNodeData::ReadFromClipboard(ui::ClipboardBuffer buffer) { |
| DCHECK_EQ(buffer, ui::ClipboardBuffer::kCopyPaste); |
| std::string data; |
| ui::Clipboard* clipboard = ui::Clipboard::GetForCurrentThread(); |
| clipboard->ReadData( |
| ui::ClipboardFormatType::CustomPlatformType(kClipboardFormatString), |
| /* data_dst = */ nullptr, &data); |
| |
| if (!data.empty()) { |
| base::Pickle pickle = |
| base::Pickle::WithUnownedBuffer(base::as_byte_span(data)); |
| if (ReadFromPickle(&pickle)) |
| return true; |
| } |
| |
| std::u16string title; |
| std::string url; |
| clipboard->ReadBookmark(/* data_dst = */ nullptr, &title, &url); |
| if (!url.empty()) { |
| Element element; |
| element.is_url = true; |
| element.url = GURL(url); |
| element.title = title; |
| |
| elements.clear(); |
| elements.push_back(element); |
| return true; |
| } |
| |
| return false; |
| } |
| |
| void BookmarkNodeData::WriteToPickle(const base::FilePath& profile_path, |
| base::Pickle* pickle) const { |
| profile_path.WriteToPickle(pickle); |
| pickle->WriteUInt32(static_cast<uint32_t>(size())); |
| |
| for (size_t i = 0; i < size(); ++i) |
| elements[i].WriteToPickle(pickle); |
| } |
| |
| bool BookmarkNodeData::ReadFromPickle(base::Pickle* pickle) { |
| base::PickleIterator data_iterator(*pickle); |
| uint32_t element_count_tmp; |
| if (profile_path_.ReadFromPickle(&data_iterator) && |
| data_iterator.ReadUInt32(&element_count_tmp)) { |
| if (!base::IsValueInRangeForNumericType<size_t>(element_count_tmp)) { |
| LOG(WARNING) << "element_count failed bounds check"; |
| return false; |
| } |
| const size_t element_count = base::checked_cast<size_t>(element_count_tmp); |
| // Restrict vector preallocation to prevent OOM crashes on invalid or |
| // malicious pickles. |
| if (element_count > kMaxVectorPreallocateSize) |
| LOG(WARNING) << "element_count exceeds kMaxVectorPreallocateSize"; |
| std::vector<Element> tmp_elements; |
| tmp_elements.reserve(std::min(element_count, kMaxVectorPreallocateSize)); |
| for (size_t i = 0; i < element_count; ++i) { |
| tmp_elements.emplace_back(); |
| if (!tmp_elements.back().ReadFromPickle(&data_iterator)) { |
| return false; |
| } |
| } |
| elements.swap(tmp_elements); |
| } |
| |
| return true; |
| } |
| |
| #endif // BUILDFLAG(IS_APPLE) |
| |
| std::vector<raw_ptr<const BookmarkNode, VectorExperimental>> |
| BookmarkNodeData::GetNodes(BookmarkModel* model, |
| const base::FilePath& profile_path) const { |
| std::vector<raw_ptr<const BookmarkNode, VectorExperimental>> nodes; |
| |
| if (!IsFromProfilePath(profile_path)) |
| return nodes; |
| |
| for (size_t i = 0; i < size(); ++i) { |
| const BookmarkNode* node = GetBookmarkNodeByID(model, elements[i].id_); |
| if (!node) { |
| nodes.clear(); |
| return nodes; |
| } |
| nodes.push_back(node); |
| } |
| return nodes; |
| } |
| |
| const BookmarkNode* BookmarkNodeData::GetFirstNode( |
| BookmarkModel* model, |
| const base::FilePath& profile_path) const { |
| std::vector<raw_ptr<const BookmarkNode, VectorExperimental>> nodes = |
| GetNodes(model, profile_path); |
| return nodes.size() == 1 ? nodes[0] : nullptr; |
| } |
| |
| void BookmarkNodeData::Clear() { |
| profile_path_.clear(); |
| elements.clear(); |
| } |
| |
| void BookmarkNodeData::SetOriginatingProfilePath( |
| const base::FilePath& profile_path) { |
| DCHECK(profile_path_.empty()); |
| profile_path_ = profile_path; |
| } |
| |
| bool BookmarkNodeData::IsFromProfilePath( |
| const base::FilePath& profile_path) const { |
| // An empty path means the data is not associated with any profile. |
| return !profile_path_.empty() && profile_path_ == profile_path; |
| } |
| |
| } // namespace bookmarks |