Files
palemoon27/toolkit/devtools/shared/node-attribute-parser.js
T
roytam1 1e24b22ef4 remove nsINetUtil_ESR_38, fixup and import changes from `dev' branch of rmottola/Arctic-Fox:
- Bug 1150366 - Remove duplicate d3.js. r=jsantell (71f991da6)
- Bug 1175352 - Refactor LoadInfo arguments to be more self contained (r=bent) (3f7d08164)
- Bug 1181533 - Add nsCRTGlue.h and prtime.h includes to nsDownloadManager.cpp to fix --disable-safe-browsing build bustage. r=froydnj (af338edf1)
- Bug 905127 - Part 1 - Make some functions from nsNetUtil not inline. r=jduell Bug 905127 - Part 2 - remove unnecessary nsNetUtil.h includes r=jduell Bug 905127 - Make some functions from nsNetUtil not inline. r=jduell (bc47d5b3a)
- Bug 1117650 - Part 1: Move all CORS tests into dom/security/test. r=sicking (6f727387c)
- Bug 1117650 - Part 2: Move all MixedContent tests into dom/security/test. r=tanvi (73f273829)
- Bug 1004703 - ignore 'unsafe-inline' if nonce- or hash-source specifed - tests (r=sstamm) (ebf411159)
- Bug 921493 - CSP: test whitelisting of scheme-relative sources (r=dveditz) (f0618b189)
- Bug 1139569 - Optimize update function for element editors in markup view;r=mratcliffe (c60d38dca)
- Partil of Bug 1139644 - Make markupview use sdk timers instead of individual timers for each window;r=pbrosset (8bc1a4d66)
- Bug 1146568 - Avoid unsafe CPOW usage warnings in browser_markupview_tag_edit_08.js; r=bgrins Making use of the devtools test frame-script to set the test node's attributes instead of using a CPOW. (61d7eed64)
- Bug 1146568 - Avoid unsafe CPOW usage in markupview tests, in assertAttributes helper; r=bgrins (efc492d5d)
- Bug 858038 - Allow moving elements in the markupview by drag/drop; r=pbrosset (51d79c15c)
- Bug 1139644 - Flash only relevant attributes in markup view when changed;r=pbrosset (29807fa46)
- Bug 1147128 - Make sure attribute shows up in markup view after removing and setting to the previous value;r=mratcliffe (1947f7417)
- Bug 1147325 - Clear box model timer on markup view destroy. r=pbrosset Prevents 'this._inspector.toolbox is null' spam in mochitest-dt (1d6808b4a)
- Bug 921102 - 1 - Linkify URIs in the inspector; r=tromey, miker This first part adds a parser for node attributes which, given some node information and an attribute name, generates a small AST-like array of objects that tells which parts of the attribute (if any) are links, and what they link to. Using this, the markup-view generates the right HTML structure to display these parts as links. This part 1 doesn't yet allow users to follow these links. (c500e270e)
- Bug 1139667 - CSP tests for fetch(). r=ckirschb (df030457f)
- Bug 1086999 - CSP: Asterisk (*) wildcard should not allow blob:, data:, or filesystem: when matching source expressions - tests (r=sstamm) (dee0f2239)
- Bug 1117650 - Part 3: Move all CSP tests into dom/security/test. r=sstamm (29c95475c)
- Bug 1146566 - 1 - Use devtools common frame-script in markupview tests and add helper; r=bgrins (b10e42468)
- Bug 1146566 - 2 - Avoid using CPOWs in test browser_markupview_tag_edit_03.js; r=bgrins This change updates the browser_markupview_tag_edit_03.js test by making use of the getDomElementInfo message listener to retrieve information about the tested nodes without having to go through CPOWs. (07f0ee3ff)
- Bug 1036324 - Adds option to walker.parents() to not traverse DocShellTreeItems of different types (24123a5d2)
- Bug 921102 - 2 - Markup-view tests for attribute links; r=miker (f6da81799)
- Bug 901250 - Add scroll into view menu item for the inspector. r=bgrins (bfc7518b1)
- Bug 994055 - Part 1: Add a toggle sidebar panel button to the inspector. r=miker This adds a new icon, next to the searchbox (like in the debugger) to expand or collapse the sidebar panel in the inspector. The state is *not* persisted yet when the toolbox is closed. The button is hidden when the toolbox host switches to 'side'. (87730e20a)
- Bug 994055 - Part 2: Add tests for the inspector sidebar toggle button. r=miker (2872c3fbd)
- Bug 921102 - 3 - Open/copy markup-view attribute links; r=bgrins This part adds contextual menu items that become enabled when the user right clicks on an attribute that has a link. Depending on the nature of the link, a new tab will be opened or a node selected. The user can also choose to copy the link in the clipboard. (9b93485f6)
- Bug 921102 - 4 - Tests for the open/copy links on markup-view attributes; r=bgrins (861ba51f6)
2021-05-21 09:14:47 +08:00

279 lines
13 KiB
JavaScript

/* -*- indent-tabs-mode: nil; js-indent-level: 2; js-indent-level: 2 -*- */
/* vim: set ft=javascript ts=2 et sw=2 tw=80: */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
/**
* This module contains a small element attribute value parser. It's primary
* goal is to extract link information from attribute values (like the href in
* <a href="/some/link.html"> for example).
*
* There are several types of linkable attribute values:
* - TYPE_URI: a URI (e.g. <a href="uri">).
* - TYPE_URI_LIST: a space separated list of URIs (e.g. <a ping="uri1 uri2">).
* - TYPE_IDREF: a reference to an other element in the same document via its id
* (e.g. <label for="input-id"> or <key command="command-id">).
* - TYPE_IDREF_LIST: a space separated list of IDREFs (e.g.
* <output for="id1 id2">).
* - TYPE_RESOURCE_URI: a URI to a javascript or css resource that can be opened
* in the devtools (e.g. <script src="uri">).
*
* parseAttribute is the parser entry function, exported on this module.
*/
const TYPE_STRING = "string";
const TYPE_URI = "uri";
const TYPE_URI_LIST = "uriList";
const TYPE_IDREF = "idref";
const TYPE_IDREF_LIST = "idrefList";
const TYPE_RESOURCE_URI = "resource";
const SVG_NS = "http://www.w3.org/2000/svg";
const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
const HTML_NS = "http://www.w3.org/1999/xhtml";
const ATTRIBUTE_TYPES = [
{namespaceURI: HTML_NS, attributeName: "action", tagName: "form", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "background", tagName: "body", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "cite", tagName: "blockquote", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "cite", tagName: "q", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "cite", tagName: "del", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "cite", tagName: "ins", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "classid", tagName: "object", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "codebase", tagName: "object", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "codebase", tagName: "applet", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "command", tagName: "menuitem", type: TYPE_IDREF},
{namespaceURI: "*", attributeName: "contextmenu", tagName: "*", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "data", tagName: "object", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "for", tagName: "label", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "for", tagName: "output", type: TYPE_IDREF_LIST},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "button", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "fieldset", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "input", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "keygen", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "label", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "object", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "output", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "select", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "form", tagName: "textarea", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "formaction", tagName: "button", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "formaction", tagName: "input", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "headers", tagName: "td", type: TYPE_IDREF_LIST},
{namespaceURI: HTML_NS, attributeName: "headers", tagName: "th", type: TYPE_IDREF_LIST},
{namespaceURI: HTML_NS, attributeName: "href", tagName: "a", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "href", tagName: "area", type: TYPE_URI},
{namespaceURI: "*", attributeName: "href", tagName: "link", type: TYPE_RESOURCE_URI,
isValid: (namespaceURI, tagName, attributes) => {
return getAttribute(attributes, "rel") === "stylesheet";
}},
{namespaceURI: "*", attributeName: "href", tagName: "link", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "href", tagName: "base", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "icon", tagName: "menuitem", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "list", tagName: "input", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "longdesc", tagName: "img", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "longdesc", tagName: "frame", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "longdesc", tagName: "iframe", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "manifest", tagName: "html", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "menu", tagName: "button", type: TYPE_IDREF},
{namespaceURI: HTML_NS, attributeName: "ping", tagName: "a", type: TYPE_URI_LIST},
{namespaceURI: HTML_NS, attributeName: "ping", tagName: "area", type: TYPE_URI_LIST},
{namespaceURI: HTML_NS, attributeName: "poster", tagName: "video", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "profile", tagName: "head", type: TYPE_URI},
{namespaceURI: "*", attributeName: "src", tagName: "script", type: TYPE_RESOURCE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "input", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "frame", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "iframe", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "img", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "audio", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "embed", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "source", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "track", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "src", tagName: "video", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "usemap", tagName: "img", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "usemap", tagName: "input", type: TYPE_URI},
{namespaceURI: HTML_NS, attributeName: "usemap", tagName: "object", type: TYPE_URI},
{namespaceURI: "*", attributeName: "xmlns", tagName: "*", type: TYPE_URI},
{namespaceURI: XUL_NS, attributeName: "command", tagName: "key", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "containment", tagName: "*", type: TYPE_URI},
{namespaceURI: XUL_NS, attributeName: "context", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "datasources", tagName: "*", type: TYPE_URI_LIST},
{namespaceURI: XUL_NS, attributeName: "insertafter", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "insertbefore", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "menu", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "observes", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "popup", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "ref", tagName: "*", type: TYPE_URI},
{namespaceURI: XUL_NS, attributeName: "removeelement", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "sortResource", tagName: "*", type: TYPE_URI},
{namespaceURI: XUL_NS, attributeName: "sortResource2", tagName: "*", type: TYPE_URI},
{namespaceURI: XUL_NS, attributeName: "src", tagName: "stringbundle", type: TYPE_URI},
{namespaceURI: XUL_NS, attributeName: "template", tagName: "*", type: TYPE_IDREF},
{namespaceURI: XUL_NS, attributeName: "tooltip", tagName: "*", type: TYPE_IDREF},
// SVG links aren't handled yet, see bug 1158831.
// {namespaceURI: SVG_NS, attributeName: "fill", tagName: "*", type: },
// {namespaceURI: SVG_NS, attributeName: "stroke", tagName: "*", type: },
// {namespaceURI: SVG_NS, attributeName: "markerstart", tagName: "*", type: },
// {namespaceURI: SVG_NS, attributeName: "markermid", tagName: "*", type: },
// {namespaceURI: SVG_NS, attributeName: "markerend", tagName: "*", type: },
// {namespaceURI: SVG_NS, attributeName: "xlink:href", tagName: "*", type: }
];
let parsers = {
[TYPE_URI]: function(attributeValue) {
return [{
type: TYPE_URI,
value: attributeValue
}];
},
[TYPE_URI_LIST]: function(attributeValue) {
let data = splitBy(attributeValue, " ");
for (let token of data) {
if (!token.type) {
token.type = TYPE_URI;
}
}
return data;
},
[TYPE_RESOURCE_URI]: function(attributeValue) {
return [{
type: TYPE_RESOURCE_URI,
value: attributeValue
}];
},
[TYPE_IDREF]: function(attributeValue) {
return [{
type: TYPE_IDREF,
value: attributeValue
}];
},
[TYPE_IDREF_LIST]: function(attributeValue) {
let data = splitBy(attributeValue, " ");
for (let token of data) {
if (!token.type) {
token.type = TYPE_IDREF;
}
}
return data;
}
};
/**
* Parse an attribute value.
* @param {String} namespaceURI The namespaceURI of the node that has the
* attribute.
* @param {String} tagName The tagName of the node that has the attribute.
* @param {Array} attributes The list of all attributes of the node. This should
* be an array of {name, value} objects.
* @param {String} attributeName The name of the attribute to parse.
* @return {Array} An array of tokens that represents the value. Each token is
* an object {type: [string|uri|resource|idref], value}.
* For instance parsing the ping attribute in <a ping="uri1 uri2"> returns:
* [
* {type: "uri", value: "uri2"},
* {type: "string", value: " "},
* {type: "uri", value: "uri1"}
* ]
*/
function parseAttribute(namespaceURI, tagName, attributes, attributeName) {
if (!hasAttribute(attributes, attributeName)) {
throw new Error(`Attribute ${attributeName} isn't part of the provided attributes`);
}
let type = getType(namespaceURI, tagName, attributes, attributeName);
if (!type) {
return [{
type: TYPE_STRING,
value: getAttribute(attributes, attributeName)
}];
}
return parsers[type](getAttribute(attributes, attributeName));
}
/**
* Get the type for links in this attribute if any.
* @param {String} namespaceURI The node's namespaceURI.
* @param {String} tagName The node's tagName.
* @param {Array} attributes The node's attributes, as a list of {name, value}
* objects.
* @param {String} attributeName The name of the attribute to get the type for.
* @return {Object} null if no type exist for this attribute on this node, the
* type object otherwise.
*/
function getType(namespaceURI, tagName, attributes, attributeName) {
for (let typeData of ATTRIBUTE_TYPES) {
let hasAttribute = attributeName === typeData.attributeName ||
typeData.attributeName === "*";
let hasNamespace = namespaceURI === typeData.namespaceURI ||
typeData.namespaceURI === "*";
let hasTagName = tagName.toLowerCase() === typeData.tagName ||
typeData.tagName === "*";
let isValid = typeData.isValid
? typeData.isValid(namespaceURI, tagName, attributes, attributeName)
: true;
if (hasAttribute && hasNamespace && hasTagName && isValid) {
return typeData.type;
}
}
return null;
}
function getAttribute(attributes, attributeName) {
for (let {name, value} of attributes) {
if (name === attributeName) {
return value;
}
}
return null;
}
function hasAttribute(attributes, attributeName) {
for (let {name, value} of attributes) {
if (name === attributeName) {
return true;
}
}
return false;
}
/**
* Split a string by a given character and return an array of objects parts.
* The array will contain objects for the split character too, marked with
* TYPE_STRING type.
* @param {String} value The string to parse.
* @param {String} splitChar A 1 length split character.
* @return {Array}
*/
function splitBy(value, splitChar) {
let data = [], i = 0, buffer = "";
while (i <= value.length) {
if (i === value.length && buffer) {
data.push({value: buffer});
}
if (value[i] === splitChar) {
if (buffer) {
data.push({value: buffer});
}
data.push({
type: TYPE_STRING,
value: splitChar
});
buffer = "";
} else {
buffer += value[i];
}
i ++;
}
return data;
}
exports.parseAttribute = parseAttribute;
// Exported for testing only.
exports.splitBy = splitBy;