@tripsnek/tmf
Version:
TypeScript Modeling Framework - A TypeScript port of the Eclipse Modeling Framework (EMF)
323 lines • 40.5 kB
JavaScript
/**
* A drop-in replacement for xml2js that converts XML to JavaScript objects
* with the same structure and behavior as xml2js.parseString()
*/
// Node type constants (in case Node global is not available)
const NODE_TYPES = {
ELEMENT_NODE: 1,
TEXT_NODE: 3,
CDATA_SECTION_NODE: 4,
};
export class XmlToJsParser {
options;
constructor(options = {}) {
this.options = {
explicitArray: options.explicitArray !== false,
mergeAttrs: options.mergeAttrs === true,
explicitRoot: options.explicitRoot !== false,
ignoreAttrs: options.ignoreAttrs === true,
trim: options.trim === true,
normalize: options.normalize === true,
...options,
};
}
/**
* Parse XML string to JavaScript object (async version)
*/
parseString(xml) {
return new Promise((resolve, reject) => {
try {
const result = this.parseStringSync(xml);
resolve(result);
}
catch (error) {
reject(error);
}
});
}
/**
* Parse XML string to JavaScript object (sync version)
*/
parseStringSync(xml) {
if (this.isDOMParserAvailable()) {
return this.parseInBrowser(xml);
}
else {
// For Node.js environment, we'll need to implement our own XML parser
// or use a lightweight alternative
return this.parseInNode(xml);
}
}
/**
* Check if DOMParser is available (browser environment)
*/
isDOMParserAvailable() {
return (typeof globalThis !== 'undefined' &&
typeof globalThis.DOMParser !== 'undefined');
}
/**
* Browser implementation using DOMParser
*/
parseInBrowser(xml) {
try {
const parser = new globalThis.DOMParser();
const xmlDoc = parser.parseFromString(xml, 'text/xml');
// Check for parser errors
const parserError = xmlDoc.querySelector('parsererror');
if (parserError) {
throw new Error(`XML Parse Error: ${parserError.textContent}`);
}
if (!xmlDoc.documentElement) {
throw new Error('Invalid XML: No document element found');
}
const result = this.domElementToJs(xmlDoc.documentElement, true); // true = isRootLevel
if (this.options.explicitRoot) {
return { [xmlDoc.documentElement.tagName]: result };
}
else {
return result;
}
}
catch (error) {
throw new Error(`XML parsing failed: ${error instanceof Error ? error.message : 'Unknown error'}`);
}
}
/**
* Node.js implementation using simple regex-based parser
* This is a lightweight XML parser that handles most common cases
*/
parseInNode(xml) {
// Remove XML declaration and comments
xml = xml.replace(/<\?xml[^>]*\?>/g, '');
xml = xml.replace(/<!--[\s\S]*?-->/g, '');
xml = xml.trim();
const stack = [];
const result = {};
let current = result;
let isRootLevel = true;
// Simple regex to match XML tags and content
const tagRegex = /<\/?([^>\s]+)([^>]*)>/g;
let lastIndex = 0;
let match;
while ((match = tagRegex.exec(xml)) !== null) {
const [fullMatch, tagName, attributesStr] = match;
const isClosing = fullMatch.startsWith('</');
const isSelfClosing = fullMatch.endsWith('/>');
// Handle text content before this tag
if (match.index > lastIndex) {
const textContent = xml.substring(lastIndex, match.index);
if (textContent.trim()) {
this.addTextContent(current, textContent);
}
}
if (isClosing) {
// Closing tag - pop from stack
if (stack.length > 0) {
current = stack.pop();
if (stack.length === 0) {
isRootLevel = true;
}
}
}
else {
// Opening tag or self-closing tag
const element = {};
// Parse attributes
if (attributesStr) {
if (!this.options.ignoreAttrs && attributesStr.trim()) {
const attrs = this.parseAttributes(attributesStr);
if (Object.keys(attrs).length > 0) {
if (this.options.mergeAttrs) {
Object.assign(element, attrs);
}
else {
element.$ = attrs;
}
}
}
}
// Add element to current parent (don't force array for root level)
if (tagName)
this.addChildElement(current, tagName, element, isRootLevel);
// If not self-closing, push to stack for children
if (!isSelfClosing) {
stack.push(current);
current = element;
isRootLevel = false;
}
}
lastIndex = tagRegex.lastIndex;
}
// Handle any remaining text content
if (lastIndex < xml.length) {
const textContent = xml.substring(lastIndex);
if (textContent.trim()) {
this.addTextContent(current, textContent);
}
}
// Return the root element
const rootKeys = Object.keys(result);
if (rootKeys.length === 1 && this.options.explicitRoot) {
return result;
}
else if (rootKeys.length === 1) {
return result[rootKeys[0]];
}
else {
return result;
}
}
/**
* Convert DOM element to JavaScript object (for browser)
*/
domElementToJs(element, isRootLevel = false) {
const result = {};
// Handle attributes
if (!this.options.ignoreAttrs &&
element.attributes &&
element.attributes.length > 0) {
const attrs = {};
for (let i = 0; i < element.attributes.length; i++) {
const attr = element.attributes[i];
attrs[attr.name] = attr.value;
}
if (this.options.mergeAttrs) {
Object.assign(result, attrs);
}
else {
result.$ = attrs;
}
}
// Group child elements by tag name
const childElements = {};
const textParts = [];
// Process all child nodes
for (let i = 0; i < element.childNodes.length; i++) {
const child = element.childNodes[i];
if (child.nodeType === NODE_TYPES.ELEMENT_NODE) {
const childElement = child;
const tagName = childElement.tagName;
if (!childElements[tagName]) {
childElements[tagName] = [];
}
childElements[tagName].push(childElement);
}
else if (child.nodeType === NODE_TYPES.TEXT_NODE ||
child.nodeType === NODE_TYPES.CDATA_SECTION_NODE) {
const textContent = child.textContent || '';
if (textContent.trim()) {
textParts.push(textContent);
}
}
}
// Add child elements to result
for (const [tagName, elements] of Object.entries(childElements)) {
const childObjects = elements.map((el) => this.domElementToJs(el, false)); // children are not root level
// Use the helper method to maintain consistency with Node.js implementation
if (elements.length === 1) {
this.addChildElement(result, tagName, childObjects[0], false);
}
else {
// Multiple elements with same tag name always become arrays
result[tagName] = childObjects;
}
}
// Handle text content
if (textParts.length > 0 && Object.keys(childElements).length === 0) {
const textContent = textParts.join('');
const finalText = this.options.trim ? textContent.trim() : textContent;
if (this.options.normalize) {
result._ = finalText.replace(/\s+/g, ' ');
}
else {
result._ = finalText;
}
}
return result;
}
/**
* Parse attribute string into object
*/
parseAttributes(attributesStr) {
const attrs = {};
const attrRegex = /(\S+)=["']([^"']*)["']/g;
let match;
while ((match = attrRegex.exec(attributesStr)) !== null) {
const [, name, value] = match;
if (name)
attrs[name] = value;
}
return attrs;
}
/**
* Add text content to current element
*/
addTextContent(current, textContent) {
const trimmed = textContent.trim();
if (trimmed) {
const finalText = this.options.trim ? trimmed : textContent;
const normalizedText = this.options.normalize
? finalText.replace(/\s+/g, ' ')
: finalText;
if (current._) {
current._ += normalizedText;
}
else {
current._ = normalizedText;
}
}
}
/**
* Add child element to parent
*/
addChildElement(parent, tagName, element, isRootLevel = false) {
if (parent[tagName]) {
// Convert to array if not already
if (!Array.isArray(parent[tagName])) {
parent[tagName] = [parent[tagName]];
}
parent[tagName].push(element);
}
else {
// For root level elements, don't force array even if explicitArray is true
// xml2js only uses arrays for child elements, not the root element
if (this.options.explicitArray && !isRootLevel) {
parent[tagName] = [element];
}
else {
parent[tagName] = element;
}
}
}
}
// Default instance with xml2js-compatible defaults
const defaultParser = new XmlToJsParser({
explicitArray: true,
mergeAttrs: false,
explicitRoot: true,
ignoreAttrs: false,
trim: false,
normalize: false,
});
/**
* Drop-in replacement for xml2js.parseString()
*/
export function parseString(xml, callback) {
if (callback) {
defaultParser
.parseString(xml)
.then((result) => callback(null, result))
.catch((err) => callback(err));
return Promise.resolve();
}
else {
return defaultParser.parseString(xml);
}
}
/**
* Synchronous version
*/
export function parseStringSync(xml) {
return defaultParser.parseStringSync(xml);
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"xml-to-js-parser.js","sourceRoot":"","sources":["../../../src/lib/ecore/xml-to-js-parser.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * A drop-in replacement for xml2js that converts XML to JavaScript objects\r\n * with the same structure and behavior as xml2js.parseString()\r\n */\r\n\r\n// Type declarations for DOM types when not available\r\ndeclare global {\r\n  interface DOMParser {\r\n    parseFromString(source: string, mimeType: string): Document;\r\n  }\r\n\r\n  interface Document {\r\n    documentElement: Element | null;\r\n    querySelector(selector: string): Element | null;\r\n  }\r\n\r\n  interface Element {\r\n    tagName: string;\r\n    attributes: NamedNodeMap;\r\n    childNodes: NodeList;\r\n    textContent: string | null;\r\n  }\r\n\r\n  interface NamedNodeMap {\r\n    length: number;\r\n    [index: number]: Attr;\r\n  }\r\n\r\n  interface Attr {\r\n    name: string;\r\n    value: string;\r\n  }\r\n\r\n  interface NodeList {\r\n    length: number;\r\n    [index: number]: Node;\r\n  }\r\n\r\n  interface Node {\r\n    nodeType: number;\r\n    textContent: string | null;\r\n  }\r\n\r\n  const Node: {\r\n    ELEMENT_NODE: number;\r\n    TEXT_NODE: number;\r\n    CDATA_SECTION_NODE: number;\r\n  };\r\n}\r\n\r\n// Node type constants (in case Node global is not available)\r\nconst NODE_TYPES = {\r\n  ELEMENT_NODE: 1,\r\n  TEXT_NODE: 3,\r\n  CDATA_SECTION_NODE: 4,\r\n} as const;\r\n\r\nexport interface XmlToJsOptions {\r\n  explicitArray?: boolean; // Default: true - always use arrays for child elements\r\n  mergeAttrs?: boolean; // Default: false - keep attributes in $ object\r\n  explicitRoot?: boolean; // Default: true - keep root element\r\n  ignoreAttrs?: boolean; // Default: false - include attributes\r\n  trim?: boolean; // Default: false - trim text content\r\n  normalize?: boolean; // Default: false - normalize whitespace\r\n}\r\n\r\nexport class XmlToJsParser {\r\n  private options: Required<XmlToJsOptions>;\r\n\r\n  constructor(options: XmlToJsOptions = {}) {\r\n    this.options = {\r\n      explicitArray: options.explicitArray !== false,\r\n      mergeAttrs: options.mergeAttrs === true,\r\n      explicitRoot: options.explicitRoot !== false,\r\n      ignoreAttrs: options.ignoreAttrs === true,\r\n      trim: options.trim === true,\r\n      normalize: options.normalize === true,\r\n      ...options,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Parse XML string to JavaScript object (async version)\r\n   */\r\n  public parseString(xml: string): Promise<any> {\r\n    return new Promise((resolve, reject) => {\r\n      try {\r\n        const result = this.parseStringSync(xml);\r\n        resolve(result);\r\n      } catch (error) {\r\n        reject(error);\r\n      }\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Parse XML string to JavaScript object (sync version)\r\n   */\r\n  public parseStringSync(xml: string): any {\r\n    if (this.isDOMParserAvailable()) {\r\n      return this.parseInBrowser(xml);\r\n    } else {\r\n      // For Node.js environment, we'll need to implement our own XML parser\r\n      // or use a lightweight alternative\r\n      return this.parseInNode(xml);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Check if DOMParser is available (browser environment)\r\n   */\r\n  private isDOMParserAvailable(): boolean {\r\n    return (\r\n      typeof globalThis !== 'undefined' &&\r\n      typeof (globalThis as any).DOMParser !== 'undefined'\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Browser implementation using DOMParser\r\n   */\r\n  private parseInBrowser(xml: string): any {\r\n    try {\r\n      const parser = new (globalThis as any).DOMParser();\r\n      const xmlDoc = parser.parseFromString(xml, 'text/xml');\r\n\r\n      // Check for parser errors\r\n      const parserError = xmlDoc.querySelector('parsererror');\r\n      if (parserError) {\r\n        throw new Error(`XML Parse Error: ${parserError.textContent}`);\r\n      }\r\n\r\n      if (!xmlDoc.documentElement) {\r\n        throw new Error('Invalid XML: No document element found');\r\n      }\r\n\r\n      const result = this.domElementToJs(xmlDoc.documentElement, true); // true = isRootLevel\r\n\r\n      if (this.options.explicitRoot) {\r\n        return { [xmlDoc.documentElement.tagName]: result };\r\n      } else {\r\n        return result;\r\n      }\r\n    } catch (error) {\r\n      throw new Error(\r\n        `XML parsing failed: ${\r\n          error instanceof Error ? error.message : 'Unknown error'\r\n        }`\r\n      );\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Node.js implementation using simple regex-based parser\r\n   * This is a lightweight XML parser that handles most common cases\r\n   */\r\n  private parseInNode(xml: string): any {\r\n    // Remove XML declaration and comments\r\n    xml = xml.replace(/<\\?xml[^>]*\\?>/g, '');\r\n    xml = xml.replace(/<!--[\\s\\S]*?-->/g, '');\r\n    xml = xml.trim();\r\n\r\n    const stack: any[] = [];\r\n    const result: any = {};\r\n    let current = result;\r\n    let isRootLevel = true;\r\n\r\n    // Simple regex to match XML tags and content\r\n    const tagRegex = /<\\/?([^>\\s]+)([^>]*)>/g;\r\n    let lastIndex = 0;\r\n    let match;\r\n\r\n    while ((match = tagRegex.exec(xml)) !== null) {\r\n      const [fullMatch, tagName, attributesStr] = match;\r\n      const isClosing = fullMatch.startsWith('</');\r\n      const isSelfClosing = fullMatch.endsWith('/>');\r\n\r\n      // Handle text content before this tag\r\n      if (match.index > lastIndex) {\r\n        const textContent = xml.substring(lastIndex, match.index);\r\n        if (textContent.trim()) {\r\n          this.addTextContent(current, textContent);\r\n        }\r\n      }\r\n\r\n      if (isClosing) {\r\n        // Closing tag - pop from stack\r\n        if (stack.length > 0) {\r\n          current = stack.pop();\r\n          if (stack.length === 0) {\r\n            isRootLevel = true;\r\n          }\r\n        }\r\n      } else {\r\n        // Opening tag or self-closing tag\r\n        const element: any = {};\r\n\r\n        // Parse attributes\r\n        if (attributesStr) {\r\n          if (!this.options.ignoreAttrs && attributesStr.trim()) {\r\n            const attrs = this.parseAttributes(attributesStr);\r\n            if (Object.keys(attrs).length > 0) {\r\n              if (this.options.mergeAttrs) {\r\n                Object.assign(element, attrs);\r\n              } else {\r\n                element.$ = attrs;\r\n              }\r\n            }\r\n          }\r\n        }\r\n\r\n        // Add element to current parent (don't force array for root level)\r\n        if (tagName)\r\n          this.addChildElement(current, tagName, element, isRootLevel);\r\n\r\n        // If not self-closing, push to stack for children\r\n        if (!isSelfClosing) {\r\n          stack.push(current);\r\n          current = element;\r\n          isRootLevel = false;\r\n        }\r\n      }\r\n\r\n      lastIndex = tagRegex.lastIndex;\r\n    }\r\n\r\n    // Handle any remaining text content\r\n    if (lastIndex < xml.length) {\r\n      const textContent = xml.substring(lastIndex);\r\n      if (textContent.trim()) {\r\n        this.addTextContent(current, textContent);\r\n      }\r\n    }\r\n\r\n    // Return the root element\r\n    const rootKeys = Object.keys(result);\r\n    if (rootKeys.length === 1 && this.options.explicitRoot) {\r\n      return result;\r\n    } else if (rootKeys.length === 1) {\r\n      return result[rootKeys[0]!];\r\n    } else {\r\n      return result;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Convert DOM element to JavaScript object (for browser)\r\n   */\r\n  private domElementToJs(element: any, isRootLevel: boolean = false): any {\r\n    const result: any = {};\r\n\r\n    // Handle attributes\r\n    if (\r\n      !this.options.ignoreAttrs &&\r\n      element.attributes &&\r\n      element.attributes.length > 0\r\n    ) {\r\n      const attrs: any = {};\r\n      for (let i = 0; i < element.attributes.length; i++) {\r\n        const attr = element.attributes[i];\r\n        attrs[attr.name] = attr.value;\r\n      }\r\n\r\n      if (this.options.mergeAttrs) {\r\n        Object.assign(result, attrs);\r\n      } else {\r\n        result.$ = attrs;\r\n      }\r\n    }\r\n\r\n    // Group child elements by tag name\r\n    const childElements: { [key: string]: any[] } = {};\r\n    const textParts: string[] = [];\r\n\r\n    // Process all child nodes\r\n    for (let i = 0; i < element.childNodes.length; i++) {\r\n      const child = element.childNodes[i];\r\n\r\n      if (child.nodeType === NODE_TYPES.ELEMENT_NODE) {\r\n        const childElement = child;\r\n        const tagName = childElement.tagName;\r\n\r\n        if (!childElements[tagName]) {\r\n          childElements[tagName] = [];\r\n        }\r\n        childElements[tagName].push(childElement);\r\n      } else if (\r\n        child.nodeType === NODE_TYPES.TEXT_NODE ||\r\n        child.nodeType === NODE_TYPES.CDATA_SECTION_NODE\r\n      ) {\r\n        const textContent = child.textContent || '';\r\n        if (textContent.trim()) {\r\n          textParts.push(textContent);\r\n        }\r\n      }\r\n    }\r\n\r\n    // Add child elements to result\r\n    for (const [tagName, elements] of Object.entries(childElements)) {\r\n      const childObjects = elements.map((el) => this.domElementToJs(el, false)); // children are not root level\r\n\r\n      // Use the helper method to maintain consistency with Node.js implementation\r\n      if (elements.length === 1) {\r\n        this.addChildElement(result, tagName, childObjects[0], false);\r\n      } else {\r\n        // Multiple elements with same tag name always become arrays\r\n        result[tagName] = childObjects;\r\n      }\r\n    }\r\n\r\n    // Handle text content\r\n    if (textParts.length > 0 && Object.keys(childElements).length === 0) {\r\n      const textContent = textParts.join('');\r\n      const finalText = this.options.trim ? textContent.trim() : textContent;\r\n      if (this.options.normalize) {\r\n        result._ = finalText.replace(/\\s+/g, ' ');\r\n      } else {\r\n        result._ = finalText;\r\n      }\r\n    }\r\n\r\n    return result;\r\n  }\r\n\r\n  /**\r\n   * Parse attribute string into object\r\n   */\r\n  private parseAttributes(attributesStr: string): any {\r\n    const attrs: any = {};\r\n    const attrRegex = /(\\S+)=[\"']([^\"']*)[\"']/g;\r\n    let match;\r\n\r\n    while ((match = attrRegex.exec(attributesStr)) !== null) {\r\n      const [, name, value] = match;\r\n      if(name)\r\n        attrs[name] = value;\r\n    }\r\n\r\n    return attrs;\r\n  }\r\n\r\n  /**\r\n   * Add text content to current element\r\n   */\r\n  private addTextContent(current: any, textContent: string): void {\r\n    const trimmed = textContent.trim();\r\n    if (trimmed) {\r\n      const finalText = this.options.trim ? trimmed : textContent;\r\n      const normalizedText = this.options.normalize\r\n        ? finalText.replace(/\\s+/g, ' ')\r\n        : finalText;\r\n\r\n      if (current._) {\r\n        current._ += normalizedText;\r\n      } else {\r\n        current._ = normalizedText;\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Add child element to parent\r\n   */\r\n  private addChildElement(\r\n    parent: any,\r\n    tagName: string,\r\n    element: any,\r\n    isRootLevel: boolean = false\r\n  ): void {\r\n    if (parent[tagName]) {\r\n      // Convert to array if not already\r\n      if (!Array.isArray(parent[tagName])) {\r\n        parent[tagName] = [parent[tagName]];\r\n      }\r\n      parent[tagName].push(element);\r\n    } else {\r\n      // For root level elements, don't force array even if explicitArray is true\r\n      // xml2js only uses arrays for child elements, not the root element\r\n      if (this.options.explicitArray && !isRootLevel) {\r\n        parent[tagName] = [element];\r\n      } else {\r\n        parent[tagName] = element;\r\n      }\r\n    }\r\n  }\r\n}\r\n\r\n// Default instance with xml2js-compatible defaults\r\nconst defaultParser = new XmlToJsParser({\r\n  explicitArray: true,\r\n  mergeAttrs: false,\r\n  explicitRoot: true,\r\n  ignoreAttrs: false,\r\n  trim: false,\r\n  normalize: false,\r\n});\r\n\r\n/**\r\n * Drop-in replacement for xml2js.parseString()\r\n */\r\nexport function parseString(\r\n  xml: string,\r\n  callback?: (err: Error | null, result?: any) => void\r\n): Promise<any> {\r\n  if (callback) {\r\n    defaultParser\r\n      .parseString(xml)\r\n      .then((result) => callback(null, result))\r\n      .catch((err) => callback(err));\r\n    return Promise.resolve();\r\n  } else {\r\n    return defaultParser.parseString(xml);\r\n  }\r\n}\r\n\r\n/**\r\n * Synchronous version\r\n */\r\nexport function parseStringSync(xml: string): any {\r\n  return defaultParser.parseStringSync(xml);\r\n}\r\n"]}