// Copyright (c) 2006-2008 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. // // How we handle the base tag better. // Current status: // At now the normal way we use to handling base tag is // a) For those links which have corresponding local saved files, such as // savable CSS, JavaScript files, they will be written to relative URLs which // point to local saved file. Why those links can not be resolved as absolute // file URLs, because if they are resolved as absolute URLs, after moving the // file location from one directory to another directory, the file URLs will // be dead links. // b) For those links which have not corresponding local saved files, such as // links in A, AREA tags, they will be resolved as absolute URLs. // c) We comment all base tags when serialzing DOM for the page. // FireFox also uses above way to handle base tag. // // Problem: // This way can not handle the following situation: // the base tag is written by JavaScript. // For example. The page "www.yahoo.com" use // "document.write(' to DOM, so all URLs which point to // local saved resource files will be resolved as // "http://www.yahoo.com/yahoo_files/...", which will cause all saved resource // files can not be loaded correctly. Also the page will be rendered ugly since // all saved sub-resource files (such as CSS, JavaScript files) and sub-frame // files can not be fetched. // Now FireFox, IE and WebKit based Browser all have this problem. // // Solution: // My solution is that we comment old base tag and write new base tag: // after the previous commented base tag. In WebKit, it // always uses the latest "href" attribute of base tag to set document's base // URL. Based on this behavior, when we encounter a base tag, we comment it and // write a new base tag after the previous commented base tag. // The new added base tag can help engine to locate correct base URL for // correctly loading local saved resource files. Also I think we need to inherit // the base target value from document object when appending new base tag. // If there are multiple base tags in original document, we will comment all old // base tags and append new base tag after each old base tag because we do not // know those old base tags are original content or added by JavaScript. If // they are added by JavaScript, it means when loading saved page, the script(s) // will still insert base tag(s) to DOM, so the new added base tag(s) can // override the incorrect base URL and make sure we alway load correct local // saved resource files. #include "config.h" #include "base/compiler_specific.h" MSVC_PUSH_WARNING_LEVEL(0); #include "DocumentType.h" #include "FrameLoader.h" #include "Document.h" #include "Element.h" #include "HTMLAllCollection.h" #include "HTMLElement.h" #include "HTMLFormElement.h" #include "HTMLMetaElement.h" #include "HTMLNames.h" #include "KURL.h" #include "markup.h" #include "PlatformString.h" #include "TextEncoding.h" MSVC_POP_WARNING(); #undef LOG #include "webkit/glue/dom_serializer.h" #include "base/string_util.h" #include "webkit/api/src/WebFrameImpl.h" #include "webkit/glue/dom_operations.h" #include "webkit/glue/dom_operations_private.h" #include "webkit/glue/dom_serializer_delegate.h" #include "webkit/glue/entity_map.h" #include "webkit/glue/glue_util.h" using WebKit::WebFrame; using WebKit::WebFrameImpl; namespace { // Default "mark of the web" declaration static const char* const kDefaultMarkOfTheWeb = "\n\n"; // Default meat content for writing correct charset declaration. static const wchar_t* const kDefaultMetaContent = L""; // Notation of start comment. static const wchar_t* const kStartCommentNotation = L""; // Default XML declaration. static const wchar_t* const kXMLDeclaration = L"\n"; // Default base tag declaration static const wchar_t* const kBaseTagDeclaration = L""; static const wchar_t* const kBaseTargetDeclaration = L" target=\"%ls\""; // Maximum length of data buffer which is used to temporary save generated // html content data. static const int kHtmlContentBufferLength = 65536; // Check whether specified unicode has corresponding html/xml entity name. // If yes, replace the character with the returned entity notation, if not // then still use original character. void ConvertCorrespondingSymbolToEntity(WebCore::String* result, const WebCore::String& value, bool in_html_doc) { unsigned len = value.length(); const UChar* start_pos = value.characters(); const UChar* cur_pos = start_pos; while (len--) { const char* entity_name = webkit_glue::EntityMap::GetEntityNameByCode(*cur_pos, in_html_doc); if (entity_name) { // Append content before entity code. if (cur_pos > start_pos) result->append(start_pos, cur_pos - start_pos); result->append("&"); result->append(entity_name); result->append(";"); start_pos = ++cur_pos; } else { cur_pos++; } } // Append the remaining content. if (cur_pos > start_pos) result->append(start_pos, cur_pos - start_pos); } } // namespace namespace webkit_glue { // SerializeDomParam Constructor. DomSerializer::SerializeDomParam::SerializeDomParam( const GURL& current_frame_gurl, const WebCore::TextEncoding& text_encoding, WebCore::Document* doc, const FilePath& directory_name) : current_frame_gurl(current_frame_gurl), text_encoding(text_encoding), doc(doc), directory_name(directory_name), has_doctype(false), has_checked_meta(false), skip_meta_element(NULL), is_in_script_or_style_tag(false), has_doc_declaration(false) { // Cache the value since we check it lots of times. is_html_document = doc->isHTMLDocument(); } // Static std::wstring DomSerializer::GenerateMetaCharsetDeclaration( const std::wstring& charset) { return StringPrintf(kDefaultMetaContent, charset.c_str()); } // Static. std::string DomSerializer::GenerateMarkOfTheWebDeclaration( const GURL& url) { return StringPrintf(kDefaultMarkOfTheWeb, url.spec().size(), url.spec().c_str()); } // Static. std::wstring DomSerializer::GenerateBaseTagDeclaration( const std::wstring& base_target) { std::wstring target_declaration = base_target.empty() ? L"" : StringPrintf(kBaseTargetDeclaration, base_target.c_str()); return StringPrintf(kBaseTagDeclaration, target_declaration.c_str()); } WebCore::String DomSerializer::PreActionBeforeSerializeOpenTag( const WebCore::Element* element, SerializeDomParam* param, bool* need_skip) { WebCore::String result; *need_skip = false; if (param->is_html_document) { // Skip the open tag of original META tag which declare charset since we // have overrided the META which have correct charset declaration after // serializing open tag of HEAD element. if (element->hasTagName(WebCore::HTMLNames::metaTag)) { const WebCore::HTMLMetaElement* meta = static_cast(element); // Check whether the META tag has declared charset or not. WebCore::String equiv = meta->httpEquiv(); if (equalIgnoringCase(equiv, "content-type")) { WebCore::String content = meta->content(); if (content.length() && content.contains("charset", false)) { // Find META tag declared charset, we need to skip it when // serializing DOM. param->skip_meta_element = element; *need_skip = true; } } } else if (element->hasTagName(WebCore::HTMLNames::htmlTag)) { // Check something before processing the open tag of HEAD element. // First we add doc type declaration if original doc has it. if (!param->has_doctype) { param->has_doctype = true; result += createMarkup(param->doc->doctype()); } // Add MOTW declaration before html tag. // See http://msdn2.microsoft.com/en-us/library/ms537628(VS.85).aspx. result += StdStringToString(GenerateMarkOfTheWebDeclaration( param->current_frame_gurl)); } else if (element->hasTagName(WebCore::HTMLNames::baseTag)) { // Comment the BASE tag when serializing dom. result += StdWStringToString(kStartCommentNotation); } } else { // Write XML declaration. if (!param->has_doc_declaration) { param->has_doc_declaration = true; // Get encoding info. WebCore::String xml_encoding = param->doc->xmlEncoding(); if (xml_encoding.isEmpty()) xml_encoding = param->doc->frame()->loader()->encoding(); if (xml_encoding.isEmpty()) xml_encoding = WebCore::UTF8Encoding().name(); std::wstring str_xml_declaration = StringPrintf(kXMLDeclaration, StringToStdWString(param->doc->xmlVersion()).c_str(), StringToStdWString(xml_encoding).c_str(), param->doc->xmlStandalone() ? L" standalone=\"yes\"" : L""); result += StdWStringToString(str_xml_declaration); } // Add doc type declaration if original doc has it. if (!param->has_doctype) { param->has_doctype = true; result += createMarkup(param->doc->doctype()); } } return result; } WebCore::String DomSerializer::PostActionAfterSerializeOpenTag( const WebCore::Element* element, SerializeDomParam* param) { WebCore::String result; param->has_added_contents_before_end = false; if (!param->is_html_document) return result; // Check after processing the open tag of HEAD element if (!param->has_checked_meta && element->hasTagName(WebCore::HTMLNames::headTag)) { param->has_checked_meta = true; // Check meta element. WebKit only pre-parse the first 512 bytes // of the document. If the whole is larger and meta is the // end of head part, then this kind of pages aren't decoded correctly // because of this issue. So when we serialize the DOM, we need to // make sure the meta will in first child of head tag. // See http://bugs.webkit.org/show_bug.cgi?id=16621. // First we generate new content for writing correct META element. std::wstring str_meta = GenerateMetaCharsetDeclaration( ASCIIToWide(param->text_encoding.name())); result += StdWStringToString(str_meta); param->has_added_contents_before_end = true; // Will search each META which has charset declaration, and skip them all // in PreActionBeforeSerializeOpenTag. } else if (element->hasTagName(WebCore::HTMLNames::scriptTag) || element->hasTagName(WebCore::HTMLNames::styleTag)) { param->is_in_script_or_style_tag = true; } return result; } WebCore::String DomSerializer::PreActionBeforeSerializeEndTag( const WebCore::Element* element, SerializeDomParam* param, bool* need_skip) { WebCore::String result; *need_skip = false; if (!param->is_html_document) return result; // Skip the end tag of original META tag which declare charset. // Need not to check whether it's META tag since we guarantee // skip_meta_element is definitely META tag if it's not NULL. if (param->skip_meta_element == element) { *need_skip = true; } else if (element->hasTagName(WebCore::HTMLNames::scriptTag) || element->hasTagName(WebCore::HTMLNames::styleTag)) { DCHECK(param->is_in_script_or_style_tag); param->is_in_script_or_style_tag = false; } return result; } // After we finish serializing end tag of a element, we give the target // element a chance to do some post work to add some additional data. WebCore::String DomSerializer::PostActionAfterSerializeEndTag( const WebCore::Element* element, SerializeDomParam* param) { WebCore::String result; if (!param->is_html_document) return result; // Comment the BASE tag when serializing DOM. if (element->hasTagName(WebCore::HTMLNames::baseTag)) { result += StdWStringToString(kEndCommentNotation); // Append a new base tag declaration. result += StdWStringToString(GenerateBaseTagDeclaration( webkit_glue::StringToStdWString(param->doc->baseTarget()))); } return result; } void DomSerializer::SaveHtmlContentToBuffer(const WebCore::String& result, SerializeDomParam* param) { if (!result.length()) return; // Convert the unicode content to target encoding WebCore::CString encoding_result = param->text_encoding.encode( result.characters(), result.length(), WebCore::EntitiesForUnencodables); // if the data buffer will be full, then send it out first. if (encoding_result.length() + data_buffer_.size() > data_buffer_.capacity()) { // Send data to delegate, tell it now we are serializing current frame. delegate_->DidSerializeDataForFrame(param->current_frame_gurl, data_buffer_, DomSerializerDelegate::CURRENT_FRAME_IS_NOT_FINISHED); data_buffer_.clear(); } // Append result to data buffer. data_buffer_.append(CStringToStdString(encoding_result)); } void DomSerializer::OpenTagToString(const WebCore::Element* element, SerializeDomParam* param) { bool need_skip; // Do pre action for open tag. WebCore::String result = PreActionBeforeSerializeOpenTag(element, param, &need_skip); if (need_skip) return; // Add open tag result += "<" + element->nodeName(); // Go through all attributes and serialize them. const WebCore::NamedNodeMap *attrMap = element->attributes(true); if (attrMap) { unsigned numAttrs = attrMap->length(); for (unsigned i = 0; i < numAttrs; i++) { result += " "; // Add attribute pair const WebCore::Attribute *attribute = attrMap->attributeItem(i); result += attribute->name().toString(); result += "=\""; if (!attribute->value().isEmpty()) { // Check whether we need to replace some resource links // with local resource paths. const WebCore::QualifiedName& attr_name = attribute->name(); // Check whether need to change the attribute which has link bool need_replace_link = ElementHasLegalLinkAttribute(element, attr_name); if (need_replace_link) { // First, get the absolute link const WebCore::String& attr_value = attribute->value(); // For links start with "javascript:", we do not change it. if (attr_value.startsWith("javascript:", false)) { result += attr_value; } else { WebCore::String str_value = param->doc->completeURL(attr_value); std::string value(StringToStdString(str_value)); // Check whether we local files for those link. LinkLocalPathMap::const_iterator it = local_links_.find(value); if (it != local_links_.end()) { // Replace the link when we have local files. FilePath::StringType path(FilePath::kCurrentDirectory); if (!param->directory_name.empty()) path += FILE_PATH_LITERAL("/") + param->directory_name.value(); path += FILE_PATH_LITERAL("/") + it->second.value(); result += FilePathStringToString(path); } else { // If not found local path, replace it with absolute link. result += str_value; } } } else { ConvertCorrespondingSymbolToEntity(&result, attribute->value(), param->is_html_document); } } result += "\""; } } // Do post action for open tag. WebCore::String added_contents = PostActionAfterSerializeOpenTag(element, param); // Complete the open tag for element when it has child/children. if (element->hasChildNodes() || param->has_added_contents_before_end) result += ">"; // Append the added contents generate in post action of open tag. result += added_contents; // Save the result to data buffer. SaveHtmlContentToBuffer(result, param); } // Serialize end tag of an specified element. void DomSerializer::EndTagToString(const WebCore::Element* element, SerializeDomParam* param) { bool need_skip; // Do pre action for end tag. WebCore::String result = PreActionBeforeSerializeEndTag(element, param, &need_skip); if (need_skip) return; // Write end tag when element has child/children. if (element->hasChildNodes() || param->has_added_contents_before_end) { result += "nodeName(); result += ">"; } else { // Check whether we have to write end tag for empty element. if (param->is_html_document) { result += ">"; const WebCore::HTMLElement* html_element = static_cast(element); if (html_element->endTagRequirement() == WebCore::TagStatusRequired) { // We need to write end tag when it is required. result += "nodeName(); result += ">"; } } else { // For xml base document. result += " />"; } } // Do post action for end tag. result += PostActionAfterSerializeEndTag(element, param); // Save the result to data buffer. SaveHtmlContentToBuffer(result, param); } void DomSerializer::BuildContentForNode(const WebCore::Node* node, SerializeDomParam* param) { switch (node->nodeType()) { case WebCore::Node::ELEMENT_NODE: { // Process open tag of element. OpenTagToString(static_cast(node), param); // Walk through the children nodes and process it. for (const WebCore::Node *child = node->firstChild(); child != NULL; child = child->nextSibling()) BuildContentForNode(child, param); // Process end tag of element. EndTagToString(static_cast(node), param); break; } case WebCore::Node::TEXT_NODE: { SaveHtmlContentToBuffer(createMarkup(node), param); break; } case WebCore::Node::ATTRIBUTE_NODE: case WebCore::Node::DOCUMENT_NODE: case WebCore::Node::DOCUMENT_FRAGMENT_NODE: { // Should not exist. DCHECK(false); break; } // Document type node can be in DOM? case WebCore::Node::DOCUMENT_TYPE_NODE: param->has_doctype = true; default: { // For other type node, call default action. SaveHtmlContentToBuffer(createMarkup(node), param); break; } } } DomSerializer::DomSerializer(WebFrame* webframe, bool recursive_serialization, DomSerializerDelegate* delegate, const std::vector& links, const std::vector& local_paths, const FilePath& local_directory_name) : delegate_(delegate), recursive_serialization_(recursive_serialization), frames_collected_(false), local_directory_name_(local_directory_name) { // Must specify available webframe. DCHECK(webframe); specified_webframeimpl_ = static_cast(webframe); // Make sure we have not-NULL delegate. DCHECK(delegate); // Build local resources map. DCHECK(links.size() == local_paths.size()); std::vector::const_iterator link_it = links.begin(); std::vector::const_iterator path_it = local_paths.begin(); for (; link_it != links.end(); ++link_it, ++path_it) { bool never_present = local_links_.insert( LinkLocalPathMap::value_type(link_it->spec(), *path_it)). second; DCHECK(never_present); } // Init data buffer. data_buffer_.reserve(kHtmlContentBufferLength); DCHECK(data_buffer_.empty()); } void DomSerializer::CollectTargetFrames() { DCHECK(!frames_collected_); frames_collected_ = true; // First, process main frame. frames_.push_back(specified_webframeimpl_); // Return now if user only needs to serialize specified frame, not including // all sub-frames. if (!recursive_serialization_) return; // Collect all frames inside the specified frame. for (int i = 0; i < static_cast(frames_.size()); ++i) { WebFrameImpl* current_frame = frames_[i]; // Get current using document. WebCore::Document* current_doc = current_frame->frame()->document(); // Go through sub-frames. RefPtr all = current_doc->all(); for (WebCore::Node* node = all->firstItem(); node != NULL; node = all->nextItem()) { if (!node->isHTMLElement()) continue; WebCore::Element* element = static_cast(node); // Check frame tag and iframe tag. bool is_frame_element; WebFrameImpl* web_frame = GetWebFrameImplFromElement( element, &is_frame_element); if (is_frame_element && web_frame) frames_.push_back(web_frame); } } } bool DomSerializer::SerializeDom() { // Collect target frames. if (!frames_collected_) CollectTargetFrames(); bool did_serialization = false; // Get GURL for main frame. GURL main_page_gurl(KURLToGURL( specified_webframeimpl_->frame()->loader()->url())); // Go through all frames for serializing DOM for whole page, include // sub-frames. for (int i = 0; i < static_cast(frames_.size()); ++i) { // Get current serializing frame. WebFrameImpl* current_frame = frames_[i]; // Get current using document. WebCore::Document* current_doc = current_frame->frame()->document(); // Get current frame's URL. const WebCore::KURL& current_frame_kurl = current_frame->frame()->loader()->url(); GURL current_frame_gurl(KURLToGURL(current_frame_kurl)); // Check whether we have done this document. if (local_links_.find(current_frame_gurl.spec()) != local_links_.end()) { // A new document, we will serialize it. did_serialization = true; // Get target encoding for current document. WebCore::String encoding = current_frame->frame()->loader()->encoding(); // Create the text encoding object with target encoding. WebCore::TextEncoding text_encoding(encoding); // Construct serialize parameter for late processing document. SerializeDomParam param( current_frame_gurl, encoding.length() ? text_encoding : WebCore::UTF8Encoding(), current_doc, current_frame_gurl == main_page_gurl ? local_directory_name_ : FilePath()); // Process current document. WebCore::Element* root_element = current_doc->documentElement(); if (root_element) BuildContentForNode(root_element, ¶m); // Sink the remainder data and finish serializing current frame. delegate_->DidSerializeDataForFrame(current_frame_gurl, data_buffer_, DomSerializerDelegate::CURRENT_FRAME_IS_FINISHED); // Clear the buffer. data_buffer_.clear(); } } // We have done call frames, so we send message to embedder to tell it that // frames are finished serializing. DCHECK(data_buffer_.empty()); delegate_->DidSerializeDataForFrame(GURL(), data_buffer_, DomSerializerDelegate::ALL_FRAMES_ARE_FINISHED); return did_serialization; } } // namespace webkit_glue