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

scala.coursier.ivy.IvyXml.scala Maven / Gradle / Ivy

package coursier.ivy

import coursier.core._
import coursier.util.Xml._
import coursier.core.Validation._

import scala.collection.compat._

object IvyXml {

  val attributesNamespace = "http://ant.apache.org/ivy/extra"

  private def info(node: Node): Either[String, (Module, String)] =
    for {
      org <- node
        .attribute("organisation")
        .flatMap(validateCoordinate(_, "organisation"))
        .map(Organization(_))
      name <- node
        .attribute("module")
        .flatMap(validateCoordinate(_, "module"))
        .map(ModuleName(_))
      version <- node
        .attribute("revision")
        .flatMap(validateCoordinate(_, "revision"))
    } yield {
      val attr = node.attributesFromNamespace(attributesNamespace)
      (Module(org, name, attr.toMap), version)
    }

  // FIXME Errors are ignored here
  private def configurations(node: Node): Seq[(Configuration, Seq[Configuration])] =
    node
      .children
      .filter(_.label == "conf")
      .flatMap { node =>
        node.attribute("name").toOption.toSeq.map(_ -> node)
      }
      .map {
        case (name, node) =>
          Configuration(name) -> node.attribute("extends").toSeq.flatMap(
            _.split(',').map(Configuration(_))
          )
      }

  // FIXME "default(compile)" likely not to be always the default
  def mappings(mapping: String): Seq[(Configuration, Configuration)] =
    mapping.split(';').toSeq.flatMap { m =>
      val (froms, tos) = m.split("->", 2) match {
        case Array(from)     => (from, Configuration.defaultCompile.value)
        case Array(from, to) => (from, to)
      }

      for {
        from <- froms.split(',').toSeq
        to   <- tos.split(',').toSeq
      } yield (Configuration(from.trim), Configuration(to.trim))
    }

  private def exclude(node0: Node): Seq[(Configuration, (Organization, ModuleName))] = {
    val org = Organization(node0.attribute("org").getOrElse("*"))
    val name = ModuleName(
      node0.attribute("module").toOption
        .orElse(node0.attribute("name").toOption)
        .getOrElse("*")
    )
    val confs = node0
      .attribute("conf")
      .toOption
      .filter(_.nonEmpty)
      .fold(Seq(Configuration.all))(_.split(',').toSeq.map(Configuration(_)))
    confs.map(_ -> (org, name))
  }

  private def override0(node0: Node): Option[(Organization, ModuleName, String)] = {
    val versionOpt = node0
      .attribute("rev")
      .toOption
      .filter(_.nonEmpty)
    versionOpt.map { version =>
      val org = Organization(node0.attribute("org").getOrElse("*"))
      val name = ModuleName(
        node0.attribute("module").toOption
          .orElse(node0.attribute("name").toOption)
          .getOrElse("*")
      )
      (org, name, version)
    }
  }

  // FIXME Errors ignored as above - warnings should be reported at least for anything suspicious
  private def dependencies(
    node: Node,
    globalExcludes: Map[Configuration, Set[(Organization, ModuleName)]],
    globalExcludesFilter: (Configuration, Organization, ModuleName) => Boolean,
    globalOverrides: DependencyManagement.Map
  ): Seq[(Configuration, Dependency)] =
    node.children
      .filter(_.label == "dependency")
      .flatMap { node =>
        // "include" sub-nodes are ignored here

        val artifacts = node
          .children
          .filter(_.label == "artifact")
          .map(publication)

        val publications =
          if (artifacts.isEmpty)
            Seq(Publication.empty)
          else
            artifacts

        val excludes = node
          .children
          .filter(_.label == "exclude")
          .flatMap(exclude)
          .groupBy(_._1)
          .view
          .mapValues(_.map(_._2).toSet)
          .toMap

        val allConfsExcludes = excludes.getOrElse(Configuration.all, Set.empty)

        val attr = node.attributesFromNamespace(attributesNamespace)
        val transitive = node.attribute("transitive") match {
          case Right("false") => false
          case _              => true
        }

        for {
          org <- node
            .attribute("org")
            .flatMap(validateCoordinate(_, "org"))
            .toOption
            .toSeq
            .map(Organization(_))
          name <- node
            .attribute("name")
            .flatMap(validateCoordinate(_, "name"))
            .toOption
            .toSeq
            .map(ModuleName(_))
          version <- node
            .attribute("rev")
            .flatMap(validateCoordinate(_, "rev"))
            .toOption
            .toSeq
          rawConf            <- node.attribute("conf").toOption.toSeq
          (fromConf, toConf) <- mappings(rawConf)
          if globalExcludesFilter(fromConf, org, name)
          pub <- publications
        } yield fromConf -> Dependency(
          Module(org, name, attr.toMap),
          version,
          toConf,
          globalExcludes.getOrElse(Configuration.all, Set.empty) ++
            globalExcludes.getOrElse(fromConf, Set.empty) ++
            allConfsExcludes ++
            excludes.getOrElse(fromConf, Set.empty),
          pub, // should come from possible artifact nodes
          optional = false,
          transitive = transitive
        ).withOverrides(globalOverrides)
      }

  private def publication(node: Node): Publication = {
    val name       = node.attribute("name").getOrElse("")
    val type0      = node.attribute("type").toOption.fold(Type.jar)(Type(_))
    val ext        = node.attribute("ext").toOption.fold(type0.asExtension)(Extension(_))
    val classifier = node.attribute("classifier").toOption.fold(Classifier.empty)(Classifier(_))
    Publication(name, type0, ext, classifier)
  }

  private def publications(node: Node): Map[Configuration, Seq[Publication]] =
    node.children
      .filter(_.label == "artifact")
      .flatMap { node =>
        val pub = publication(node)
        val confs = node
          .attribute("conf")
          .fold(_ => Seq(Configuration.all), _.split(',').toSeq.map(Configuration(_)))
        confs.map(_ -> pub)
      }
      .groupBy { case (conf, _) => conf }
      .map { case (conf, l) => conf -> l.map { case (_, p) => p } }

  def project(node: Node): Either[String, Project] =
    for {
      infoNode <- node.children
        .find(_.label == "info")
        .toRight("Info not found")

      modVer <- info(infoNode)
    } yield {

      val (module0, version) = modVer
      val (extraInfo, attr) = module0.attributes
        .partition(_._1.startsWith("info."))
      val module =
        if (extraInfo.isEmpty) module0
        else module0.withAttributes(attr)

      val dependenciesNodeOpt = node.children
        .find(_.label == "dependencies")

      val globalExcludes = dependenciesNodeOpt
        .map(_.children)
        .getOrElse(Nil)
        .filter(_.label == "exclude")
        .flatMap(exclude)
        .groupBy(_._1)
        .view
        .mapValues(_.map(_._2).toSet)
        .toMap
      val filter = {
        val filters       = globalExcludes.view.mapValues(set => MinimizedExclusions(set)).toMap
        val allConfFilter = filters.get(Configuration.all)
        (conf: Configuration, org: Organization, name: ModuleName) =>
          allConfFilter.forall(_(org, name)) && {
            val confFilter = filters.get(conf)
            confFilter.forall(_(org, name))
          }
      }

      // https://ant.apache.org/ivy/history/2.5.0-rc1/ivyfile/override.html
      val globalOverrides = dependenciesNodeOpt
        .map(_.children)
        .getOrElse(Nil)
        .filter(_.label == "override")
        .flatMap(override0)
        .map {
          case (org, name, ver) =>
            DependencyManagement.Key(org, name, Type.jar, Classifier.empty) ->
              DependencyManagement.Values(
                Configuration.empty,
                ver,
                MinimizedExclusions.zero,
                optional = false
              )
        }
        .toMap
      val dependencies0 = dependenciesNodeOpt
        .map(dependencies(_, globalExcludes, filter, globalOverrides))
        .getOrElse(Nil)

      val configurationsNodeOpt = node.children
        .find(_.label == "configurations")

      val configurationsOpt = configurationsNodeOpt.map(configurations)

      val configurations0 =
        configurationsOpt.getOrElse(Seq(Configuration.default -> Seq.empty[Configuration]))

      val publicationsNodeOpt = node.children
        .find(_.label == "publications")

      val publicationsOpt = publicationsNodeOpt.map(publications)

      val descriptionNodeOpt = infoNode.children.find(_.label == "description")

      val description = descriptionNodeOpt
        .map(_.textContent.trim)
        .getOrElse("")

      val homePage = descriptionNodeOpt.flatMap(_.attribute("homepage").toOption).getOrElse("")

      val licenses = infoNode.children
        .filter(_.label == "license")
        .flatMap { n =>
          n.attribute("name").toSeq.map { name =>
            (name, n.attribute("url").toOption)
          }
        }

      val publicationDate = infoNode.attribute("publication").toOption.flatMap(parseDateTime)

      Project(
        module,
        version,
        dependencies0,
        configurations0.toMap,
        parent = None,
        dependencyManagement = Nil,
        properties = extraInfo.toSeq,
        profiles = Nil,
        versions = None,
        snapshotVersioning = None,
        packagingOpt = None,
        relocated = false,
        actualVersionOpt = None,
        if (publicationsOpt.isEmpty)
          // no publications node -> default JAR artifact
          Seq(Configuration.all -> Publication(
            module.name.value,
            Type.jar,
            Extension.jar,
            Classifier.empty
          ))
        else {
          // publications node is there -> only its content (if it is empty, no artifacts,
          // as per the Ivy manual)
          val inAllConfs = publicationsOpt.flatMap(_.get(Configuration.all)).getOrElse(Nil)
          configurations0.flatMap { case (conf, _) =>
            (publicationsOpt.flatMap(_.get(conf)).getOrElse(Nil) ++ inAllConfs).map(conf -> _)
          }
        },
        Info(
          description,
          homePage,
          licenses,
          Nil,
          publicationDate,
          None
        )
      )
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy