com.nawforce.runtime.xml.XMLDocument.scala Maven / Gradle / Ivy
/*
Copyright (c) 2019 Kevin Jones, All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. The name of the author may not be used to endorse or promote products
derived from this software without specific prior written permission.
*/
package com.nawforce.runtime.xml
import com.nawforce.pkgforce.diagnostics._
import com.nawforce.pkgforce.path.{Location, PathLike}
import com.nawforce.pkgforce.xml.{XMLDocumentLike, XMLElementLike, XMLName}
import com.nawforce.runtime.parsers.SourceData
import scala.collection.immutable.ArraySeq
import scala.collection.mutable.ArrayBuffer
import scala.scalajs.js
import scala.scalajs.js.{Dynamic, Object}
import scala.util.matching.Regex
class XMLElement(element: Element) extends XMLElementLike {
override lazy val line: Int = element.lineNumber
override lazy val name: XMLName = XMLName(element.namespaceURI, element.localName)
override lazy val text: String = {
val nl = element.childNodes
val sb = new StringBuilder()
for (i <- 0 until nl.length) {
val n = nl.item(i)
if (n.nodeType == Node.TEXT_NODE) {
sb.append(n.asInstanceOf[Text].data)
}
}
sb.toString()
}
override def getChildren(name: String): Seq[XMLElementLike] = {
val matched = ArrayBuffer[XMLElementLike]()
val nl = element.childNodes
for (i <- 0 until nl.length) {
val n = nl.item(i)
if (
n.nodeType == Node.ELEMENT_NODE && n.namespaceURI == XMLDocument.sfNamespace && n.localName == name
) {
matched.append(new XMLElement(n.asInstanceOf[Element]))
}
}
matched.toSeq
}
}
class XMLDocument(path: PathLike, doc: Document) extends XMLDocumentLike(path) {
override lazy val rootElement: XMLElementLike = new XMLElement(doc.documentElement)
}
object XMLDocument {
val sfNamespace = "http://soap.sforce.com/2006/04/metadata"
var errors: List[Issue] = Nil
def apply(path: PathLike, sourceData: SourceData): IssuesAnd[Option[XMLDocument]] = {
errors = Nil
val parser = new DOMParser(getOptions(path))
val doc = parser.parseFromString(sourceData.asString, "text/xml")
if (errors.nonEmpty) {
IssuesAnd(ArraySeq(errors.last), None)
} else {
IssuesAnd(Some(new XMLDocument(path, doc)))
}
}
private def getOptions(path: PathLike): Object with Dynamic = {
js.Dynamic.literal(
"locator" -> js.Dynamic.literal(),
"errorHandler" ->
js.Dynamic.literal(
"warning" -> { msg: String =>
captureErrors(path, ERROR_CATEGORY, msg)
},
"error" -> { msg: String =>
captureErrors(path, ERROR_CATEGORY, msg)
},
"fatalError" -> { msg: String =>
captureErrors(path, ERROR_CATEGORY, msg)
}
)
)
}
private def captureErrors(path: PathLike, category: DiagnosticCategory, msg: String): Unit = {
errors = toError(path, category, msg) :: errors
}
private val lineMatch: Regex = "line:[0-9]*".r
private val columnMatch: Regex = "col:[0-9]*".r
private def toError(path: PathLike, category: DiagnosticCategory, msg: String): Issue = {
val line = lineMatch.findFirstIn(msg) match {
case Some(l) =>
try {
l.replaceFirst("line:", "").toInt
} catch {
case _: NumberFormatException => 0
}
case None => 0
}
val column = columnMatch.findFirstIn(msg) match {
case Some(l) =>
try {
l.replaceFirst("col:", "").toInt
} catch {
case _: NumberFormatException => 0
}
case None => 0
}
val trimmedMsg =
if (line == 0 || column == 0)
msg.split("@#\\[")(0)
else
msg
Issue(path, Diagnostic(category, Location(Math.max(1, line), column), trimmedMsg))
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy