aboutsummaryrefslogtreecommitdiffstats
path: root/vanilla/node_modules/w3c-xmlserializer/lib/attributes.js
diff options
context:
space:
mode:
authorAdam Mathes <adam@adammathes.com>2026-02-13 21:34:48 -0800
committerAdam Mathes <adam@adammathes.com>2026-02-13 21:34:48 -0800
commit76cb9c2a39d477a64824a985ade40507e3bbade1 (patch)
tree41e997aa9c6f538d3a136af61dae9424db2005a9 /vanilla/node_modules/w3c-xmlserializer/lib/attributes.js
parent819a39a21ac992b1393244a4c283bbb125208c69 (diff)
downloadneko-76cb9c2a39d477a64824a985ade40507e3bbade1.tar.gz
neko-76cb9c2a39d477a64824a985ade40507e3bbade1.tar.bz2
neko-76cb9c2a39d477a64824a985ade40507e3bbade1.zip
feat(vanilla): add testing infrastructure and tests (NK-wjnczv)
Diffstat (limited to 'vanilla/node_modules/w3c-xmlserializer/lib/attributes.js')
-rw-r--r--vanilla/node_modules/w3c-xmlserializer/lib/attributes.js125
1 files changed, 125 insertions, 0 deletions
diff --git a/vanilla/node_modules/w3c-xmlserializer/lib/attributes.js b/vanilla/node_modules/w3c-xmlserializer/lib/attributes.js
new file mode 100644
index 0000000..5012e5b
--- /dev/null
+++ b/vanilla/node_modules/w3c-xmlserializer/lib/attributes.js
@@ -0,0 +1,125 @@
+"use strict";
+
+const xnv = require("xml-name-validator");
+
+const { NAMESPACES } = require("./constants");
+
+function generatePrefix(map, newNamespace, prefixIndex) {
+ const generatedPrefix = `ns${prefixIndex}`;
+ map[newNamespace] = [generatedPrefix];
+ return generatedPrefix;
+}
+
+function preferredPrefixString(map, ns, preferredPrefix) {
+ const candidateList = map[ns];
+ if (!candidateList) {
+ return null;
+ }
+ if (candidateList.includes(preferredPrefix)) {
+ return preferredPrefix;
+ }
+ return candidateList[candidateList.length - 1];
+}
+
+function serializeAttributeValue(value/* , requireWellFormed*/) {
+ if (value === null) {
+ return "";
+ }
+ // TODO: Check well-formedness
+ return value
+ .replace(/&/ug, "&amp;")
+ .replace(/"/ug, "&quot;")
+ .replace(/</ug, "&lt;")
+ .replace(/>/ug, "&gt;")
+ .replace(/\t/ug, "&#x9;")
+ .replace(/\n/ug, "&#xA;")
+ .replace(/\r/ug, "&#xD;");
+}
+
+function serializeAttributes(
+ element,
+ map,
+ localPrefixes,
+ ignoreNamespaceDefAttr,
+ requireWellFormed,
+ refs
+) {
+ let result = "";
+ const namespaceLocalnames = Object.create(null);
+ for (const attr of element.attributes) {
+ if (
+ requireWellFormed &&
+ namespaceLocalnames[attr.namespaceURI] &&
+ namespaceLocalnames[attr.namespaceURI].has(attr.localName)
+ ) {
+ throw new Error("Found duplicated attribute");
+ }
+ if (!namespaceLocalnames[attr.namespaceURI]) {
+ namespaceLocalnames[attr.namespaceURI] = new Set();
+ }
+ namespaceLocalnames[attr.namespaceURI].add(attr.localName);
+ const attributeNamespace = attr.namespaceURI;
+ let candidatePrefix = null;
+ if (attributeNamespace !== null) {
+ candidatePrefix = preferredPrefixString(
+ map,
+ attributeNamespace,
+ attr.prefix
+ );
+ if (attributeNamespace === NAMESPACES.XMLNS) {
+ if (
+ attr.value === NAMESPACES.XML ||
+ (attr.prefix === null && ignoreNamespaceDefAttr) ||
+ (attr.prefix !== null &&
+ localPrefixes[attr.localName] !== attr.value &&
+ map[attr.value].includes(attr.localName))
+ ) {
+ continue;
+ }
+ if (requireWellFormed && attr.value === NAMESPACES.XMLNS) {
+ throw new Error(
+ "The XMLNS namespace is reserved and cannot be applied as an element's namespace via XML parsing"
+ );
+ }
+ if (requireWellFormed && attr.value === "") {
+ throw new Error(
+ "Namespace prefix declarations cannot be used to undeclare a namespace"
+ );
+ }
+ if (attr.prefix === "xmlns") {
+ candidatePrefix = "xmlns";
+ }
+ } else if (candidatePrefix === null) {
+ candidatePrefix = generatePrefix(
+ map,
+ attributeNamespace,
+ refs.prefixIndex++
+ );
+ result += ` xmlns:${candidatePrefix}="${serializeAttributeValue(
+ attributeNamespace,
+ requireWellFormed
+ )}"`;
+ }
+ }
+
+ result += " ";
+ if (candidatePrefix !== null) {
+ result += `${candidatePrefix}:`;
+ }
+ if (
+ requireWellFormed &&
+ (attr.localName.includes(":") ||
+ !xnv.name(attr.localName) ||
+ (attr.localName === "xmlns" && attributeNamespace === null))
+ ) {
+ throw new Error("Invalid attribute localName value");
+ }
+ result += `${attr.localName}="${serializeAttributeValue(attr.value, requireWellFormed)}"`;
+ }
+ return result;
+}
+
+module.exports.preferredPrefixString = preferredPrefixString;
+module.exports.generatePrefix = generatePrefix;
+module.exports.serializeAttributeValue = serializeAttributeValue;
+module.exports.serializeAttributes = serializeAttributes;