All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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