Source: converter.mjs

import { readFile } from "node:fs";
import { promisify } from "node:util";

import { DOMParser } from "common-xml-features";

/**
 * @class Class for parsing a XML file and extracting a schematic.
 * This static class contains many helper functions to extract information from the file. The extracted node of the
 * schematic view can than be passed to `Schematic.fromXML`.
 */
class Converter {
	/**
	 * Parses a file using its file handle. The file will be completely read and a list of cells will be returned.
	 *
	 * The file won't be closed.
	 *
	 * @param {number} inFileDescriptor - the file descriptor for the file to parse, e.g. 0 for stdin
	 * @param {string} cellName - the name of the cell to parse
	 * @param {number} [scale=2.54] - factor for scaling; ADS uses inch, TikZ uses cm, thus scale=2.54 is recommended
	 * @returns {Element[]} an array of cells
	 * @throws {Error} if an expected xml-tag does not exist
	 */
	static async parseFile(inFileDescriptor) {
		const promisifiedRead = promisify(readFile);

		let buffer = await promisifiedRead(inFileDescriptor).then((buff) =>
			buff instanceof Buffer ? buff.toString("utf-8") : buff
		);

		/** @type {DOMParser} */
		const parser = new DOMParser();
		/** @type {XMLDocument} */
		let parsed;
		try {
			parsed = parser.parseFromString(buffer, "text/xml");
		} catch (_error) {
			throw new Error("Syntax error in XML file.");
		}
		/** @type {Element} */
		const ABLRoot = parsed.documentElement;
		this.assertTagFound(ABLRoot, "The XML root tag");
		const Library = this.getNamedTag(ABLRoot, "library");
		this.assertTagFound(Library, "library");
		const Cells = this.getNamedTag(Library, "cells");
		this.assertTagFound(Cells, "cells");

		const CellArray = this.getNamedTags(Cells, "cell");
		return CellArray;
	}

	/**
	 * Filters a list of cell nodes by name.
	 *
	 * @param {Element[]} cellArray - the array of cells obtained from {@link parseFile}
	 * @param {string} cellname - the name of the cell to parse or `""` to use the first one
	 * @throws {Error} - if cell was not found
	 * @returns {Element} the wanted cell node
	 */
	static findCell(cellArray, cellname) {
		let thisCell;
		if (cellname == "") {
			thisCell = cellArray[0];
			if (!thisCell) throw new Error("No cells found");
		} else {
			thisCell = cellArray.find((node) => node.getAttribute("name") == cellname);
			if (!thisCell) throw new Error('Cell "' + cellname + '" not found');
		}
		return thisCell;
	}

	/**
	 * Finds the `schematicview`s of a cell.
	 *
	 * @param {Element} cell - the cell node to find the schematics in
	 * @returns {Element[]} an array of `schematicview`s
	 */
	static getSchematicViews(cell) {
		const views = this.getNamedTag(cell, "views");
		if (!views) return [];
		const schematicViewArray = this.getNamedTags(
			views,
			"schematicview",
			(node) => node.getAttribute("type") == "schematic"
		);
		return schematicViewArray;
	}

	/**
	 * Find a specific schematic in a list.
	 *
	 * @param {Element[]} schematicViewArray - the list of nodes to search in
	 * @param {string} schematicViewName - the name of the schematic or `""` to use the first one
	 * @throws {Error} - if the desired schematic can not be found
	 * @returns {Element} the found node
	 */
	static findSchematicView(schematicViewArray, schematicViewName) {
		let thisSchematicView;
		if (schematicViewName == "") {
			thisSchematicView = schematicViewArray[0];
			if (!thisSchematicView) throw new Error("Error: No schematic found");
		} else {
			thisSchematicView = Array.prototype.find.call(
				schematicViewArray,
				(node) => node.getAttribute("name") == schematicViewName
			);
			if (!thisSchematicView) throw new Error('Error: Schematic "' + schematicViewName + '" not found');
		}
		return thisSchematicView;
	}

	/**
	 * Prints the names of found cells or schematics.
	 *
	 * @param {Element[]} schematicViewArray - the list of nodes to search in
	 * @param {string} [heading] - heading to print
	 */
	static printNodeList(schematicViewArray, heading) {
		if (schematicViewArray && schematicViewArray.length > 0) {
			if (heading) console.log(heading);
			schematicViewArray.forEach((node) => console.log(" - " + (node.getAttribute("name") || "- unnamed -")));
		} else console.log((heading || "") + "none found.");
	}

	//-- Helper functions
	/**
	 * Searches a child node by name and by an additional filter if present.
	 *
	 * @private
	 * @param {Element} root - the root node to find the child
	 * @param {string} tagName  the name of the xml tag to find
	 * @param {function(Element): boolean} [additionalFilter] - filter function returning true if node matches the criteria
	 * @returns {Element|null} the found node or null if not found
	 */
	static getNamedTag(root, tagName, additionalFilter) {
		return Array.prototype.find.call(
			/** @type {NodeList} */ root.childNodes,
			(node) =>
				node.nodeType === 1 && // node instanceof Element
				node.localName &&
				node.localName.toLowerCase() == tagName &&
				(!additionalFilter || additionalFilter(node))
		);
	}

	/**
	 * Filters child nodes by name and by an additional filter if present.
	 *
	 * @private
	 * @param {Element} root - the root node to find the children
	 * @param {string} tagName - the name of the xml tag to filter
	 * @param {function(Element): boolean} [additionalFilter] - filter function returning true if node matches the criteria
	 * @returns {Element[]} the filtered nodes (may be empty)
	 */
	static getNamedTags(root, tagName, additionalFilter) {
		return Array.prototype.filter.call(
			/** @type {NodeList} */ root.childNodes,
			(node) =>
				node.nodeType === 1 && // node instanceof Element
				node.localName &&
				node.localName.toLowerCase() == tagName &&
				(!additionalFilter || additionalFilter(node))
		);
	}

	/**
	 * Internal helper function for checking if a variable is correctly set.
	 *
	 * The message of the thrown error depends on the data type.
	 * If an variable is falsy (`null`, `undefined`, etc.) a error with the message "XML-Tag not found: \<tagname\>" is
	 * thrown. If the variable is an empty array, the message is instead "Filtered list of tags is empty: \<tagname\>".
	 *
	 * @param {*} variable - the variable to check
	 * @param {string} tagname - the name of the variable for the error message
	 * @throws {Error} the Error if `variable` is falsy or an empty array
	 */
	static assertTagFound(variable, tagname) {
		if (!variable) throw new Error("XML-Tag not found: " + tagname);
		else if (Array.isArray(variable) && variable.length === 0)
			throw new Error("Filtered list of tags is empty: " + tagname);
	}
}

export { Converter };