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

scala.coursier.maven.PomParser.scala Maven / Gradle / Ivy

There is a newer version: 2.1.22
Show newest version
package coursier.maven

import coursier.core._
import coursier.core.Validation._
import coursier.util.SaxHandler

import scala.collection.mutable.ListBuffer

final class PomParser extends SaxHandler {

  import PomParser._

  private[this] val state = new State

  private[this] var paths: CustomList[String] = CustomList.Nil
  private[this] var handlers                  = List.empty[Option[Handler]]

  private[this] val b = new java.lang.StringBuilder

  def startElement(tagName: String): Unit = {
    paths = tagName :: paths
    val handlerOpt = handlerMap.get(paths).orElse(handlerMap.get("*" :: paths.tail))
    handlers = handlerOpt :: handlers
    b.setLength(0)

    handlerOpt.foreach {
      case _: ContentHandler =>
      case s: SectionHandler =>
        s.start(state)
      case p: PropertyHandler =>
        p.name(state, tagName)
    }
  }
  def characters(ch: Array[Char], start: Int, length: Int): Unit = {
    val readContent = handlers.headOption.exists(_.exists {
      case _: PropertyHandler => true
      case _: ContentHandler  => true
      case _: SectionHandler  => false
    })
    if (readContent)
      b.append(ch, start, length)
  }
  def endElement(tagName: String): Unit = {
    val handlerOpt = handlers.headOption.flatten
    paths = paths.tail
    handlers = handlers.tail

    handlerOpt.foreach {
      case p: PropertyHandler =>
        p.content(state, b.toString)
      case c: ContentHandler =>
        c.content(state, b.toString.trim)
      case s: SectionHandler =>
        s.end(state)
    }

    b.setLength(0)
  }

  def project: Either[String, Project] =
    state.project
}

object PomParser {

  private sealed abstract class CustomList[+T] {
    def ::[U >: T](elem: U): CustomList[U] =
      CustomList.Cons(elem, this)
    def tail: CustomList[T]
  }

  private object CustomList {
    case object Nil extends CustomList[Nothing] {
      def tail: CustomList[Nothing] =
        throw new NoSuchElementException("tail of empty list")
    }
    final case class Cons[+T](head: T, override val tail: CustomList[T]) extends CustomList[T] {
      override val hashCode: Int =
        31 * (31 + head.hashCode) + tail.hashCode
    }
    def apply[T](l: List[T]): CustomList[T] =
      l match {
        case scala.Nil => CustomList.Nil
        case h :: t    => CustomList.Cons(h, CustomList(t))
      }
  }

  private final class State {
    var groupId       = ""
    var artifactIdOpt = Option.empty[String]
    var version       = ""

    var parentGroupIdOpt    = Option.empty[String]
    var parentArtifactIdOpt = Option.empty[String]
    var parentVersion       = ""

    var description            = ""
    var url                    = ""
    val licenseInfo            = new ListBuffer[Info.License]
    var licenseName            = ""
    var licenseUrl             = Option.empty[String]
    var licenseDistribution    = Option.empty[String]
    var licenseComments        = Option.empty[String]
    val developers             = Nil                             // TODO
    val publication            = Option.empty[Versions.DateTime] // TODO
    var scmOpt                 = Option.empty[Info.Scm]
    var scmUrl                 = Option.empty[String]
    var scmConnection          = Option.empty[String]
    var scmDeveloperConnection = Option.empty[String]

    var packagingOpt = Option.empty[Type]

    val dependencies         = new ListBuffer[(Configuration, Dependency)]
    val dependencyManagement = new ListBuffer[(Configuration, Dependency)]

    val properties = new ListBuffer[(String, String)]

    var relocationGroupIdOpt    = Option.empty[Organization]
    var relocationArtifactIdOpt = Option.empty[ModuleName]
    var relocationVersionOpt    = Option.empty[String]

    var dependencyGroupIdOpt    = Option.empty[Organization]
    var dependencyArtifactIdOpt = Option.empty[ModuleName]
    var dependencyVersion       = ""
    var dependencyOptional      = false
    var dependencyScope         = Configuration.empty
    var dependencyType          = Type.empty
    var dependencyClassifier    = Classifier.empty
    var dependencyExclusions    = Set.empty[(Organization, ModuleName)]

    var dependencyExclusionGroupId    = Organization("*")
    var dependencyExclusionArtifactId = ModuleName("*")

    var propertyNameOpt = Option.empty[String]

    var profileId                     = ""
    val profileDependencies           = new ListBuffer[(Configuration, Dependency)]
    val profileDependencyManagement   = new ListBuffer[(Configuration, Dependency)]
    var profileProperties             = Map.empty[String, String]
    val profileActivationProperties   = new ListBuffer[(String, Option[String])]
    var profileActiveByDefaultOpt     = Option.empty[Boolean]
    var profilePropertyNameOpt        = Option.empty[String]
    var profilePropertyValueOpt       = Option.empty[String]
    var profileActivationOsArchOpt    = Option.empty[String]
    var profileActivationOsFamilyOpt  = Option.empty[String]
    var profileActivationOsNameOpt    = Option.empty[String]
    var profileActivationOsVersionOpt = Option.empty[String]
    var profileActivationJdkOpt       = Option.empty[Either[VersionInterval, Seq[Version]]]

    val profiles = new ListBuffer[Profile]

    def project: Either[String, Project] = {

      val groupIdOpt = Some(groupId).filter(_.nonEmpty)
        .orElse(parentGroupIdOpt.filter(_.nonEmpty))

      val versionOpt = Some(version).filter(_.nonEmpty)
        .orElse(Some(parentVersion).filter(_.nonEmpty))

      val properties0 = properties.toList

      val parentModuleOpt =
        for {
          parentGroupId <- parentGroupIdOpt
            .toRight("Parent organization missing")
            .flatMap(validateCoordinate(_, "parent groupId"))
          parentArtifactId <- parentArtifactIdOpt
            .toRight("Parent artifactId missing")
            .flatMap(validateCoordinate(_, "parent artifactId"))
        } yield Module(Organization(parentGroupId), ModuleName(parentArtifactId), Map.empty)

      for {
        finalGroupId <- groupIdOpt
          .toRight("No organization found")
          .flatMap(validateCoordinate(_, "groupId"))
        artifactId <- artifactIdOpt
          .toRight("No artifactId found")
          .flatMap(validateCoordinate(_, "artifactId"))
        finalVersion <- versionOpt
          .toRight("No version found")
          .flatMap(validateCoordinate(_, "version"))

        _ <- {
          if (parentModuleOpt.exists(_.organization.value.isEmpty))
            Left("Parent organization missing")
          else
            Right(())
        }

        _ <- {
          if (parentModuleOpt.isRight && parentVersion.isEmpty)
            Left("No parent version found")
          else
            Right(())
        }

      } yield {

        val parentOpt = for {
          parentModule  <- parentModuleOpt
          parentVersion <- validateCoordinate(parentVersion, "parent version")
        } yield (parentModule, parentVersion)

        val projModule = Module(Organization(finalGroupId), ModuleName(artifactId), Map.empty)

        val relocationDependencyOpt = {
          val isRelocated = relocationGroupIdOpt.nonEmpty ||
            relocationArtifactIdOpt.nonEmpty ||
            relocationVersionOpt.nonEmpty
          if (isRelocated)
            Some {
              Configuration.empty -> Dependency(
                Module(
                  organization = relocationGroupIdOpt.getOrElse(projModule.organization),
                  name = relocationArtifactIdOpt.getOrElse(projModule.name),
                  attributes = projModule.attributes
                ),
                relocationVersionOpt.getOrElse(finalVersion),
                Configuration.empty,
                Set.empty[(Organization, ModuleName)],
                Attributes.empty,
                optional = false,
                transitive = true
              )
            }
          else
            None
        }

        Project(
          projModule,
          finalVersion,
          relocationDependencyOpt.toList ::: dependencies.toList,
          Map.empty,
          parentOpt.toOption,
          dependencyManagement.toList,
          properties0,
          profiles.toList,
          None,
          None,
          packagingOpt,
          relocated = relocationDependencyOpt.nonEmpty,
          None,
          Nil,
          Info(
            description,
            url,
            developers,
            publication,
            scmOpt,
            licenseInfo.toSeq
          )
        )
      }
    }
  }

  private sealed abstract class Handler(val path: List[String])

  private abstract class SectionHandler(path: List[String]) extends Handler(path) {
    def start(state: State): Unit
    def end(state: State): Unit
  }

  private abstract class ContentHandler(path: List[String]) extends Handler(path) {
    def content(state: State, content: String): Unit
  }

  private abstract class PropertyHandler(path: List[String]) extends Handler("*" :: path) {
    def name(state: State, name: String): Unit
    def content(state: State, content: String): Unit
  }

  private def content(path: List[String])(f: (State, String) => Unit): Handler =
    new ContentHandler(path) {
      def content(state: State, content: String) =
        f(state, content)
    }

  private val handlers = Seq[Handler](
    content("groupId" :: "project" :: Nil) {
      (state, content) =>
        state.groupId = content
    },
    content("artifactId" :: "project" :: Nil) {
      (state, content) =>
        state.artifactIdOpt = Some(content)
    },
    content("version" :: "project" :: Nil) {
      (state, content) =>
        state.version = content
    },
    content("groupId" :: "parent" :: "project" :: Nil) {
      (state, content) =>
        state.parentGroupIdOpt = Some(content)
    },
    content("artifactId" :: "parent" :: "project" :: Nil) {
      (state, content) =>
        state.parentArtifactIdOpt = Some(content)
    },
    content("version" :: "parent" :: "project" :: Nil) {
      (state, content) =>
        state.parentVersion = content
    },
    content("description" :: "project" :: Nil) {
      (state, content) =>
        state.description = content
    },
    content("url" :: "project" :: Nil) {
      (state, content) =>
        state.url = content
    },
    content("packaging" :: "project" :: Nil) {
      (state, content) =>
        state.packagingOpt = Some(Type(content))
    },
    content("groupId" :: "relocation" :: "distributionManagement" :: "project" :: Nil) {
      (state, content) =>
        state.relocationGroupIdOpt = Some(Organization(content))
    },
    content("artifactId" :: "relocation" :: "distributionManagement" :: "project" :: Nil) {
      (state, content) =>
        state.relocationArtifactIdOpt = Some(ModuleName(content))
    },
    content("version" :: "relocation" :: "distributionManagement" :: "project" :: Nil) {
      (state, content) =>
        state.relocationVersionOpt = Some(content)
    }
  ) ++ dependencyHandlers(
    "dependency" :: "dependencies" :: "project" :: Nil,
    (s, c, d) =>
      s.dependencies += c -> d
  ) ++ dependencyHandlers(
    "dependency" :: "dependencies" :: "dependencyManagement" :: "project" :: Nil,
    (s, c, d) =>
      s.dependencyManagement += c -> d
  ) ++ propertyHandlers(
    "properties" :: "project" :: Nil,
    (s, k, v) =>
      s.properties += k -> v
  ) ++ profileHandlers(
    "profile" :: "profiles" :: "project" :: Nil,
    (s, p) =>
      s.profiles += p
  ) ++ scmHandlers(
    "scm" :: "project" :: Nil,
    (s, scm) =>
      s.scmOpt = Some(scm)
  ) ++ licenseHandlers(
    "license" :: "licenses" :: "project" :: Nil,
    (s, l) =>
      s.licenseInfo += l
  )

  private def profileHandlers(prefix: List[String], add: (State, Profile) => Unit) =
    Seq(
      new SectionHandler(prefix) {
        def start(state: State) = {
          state.profileId = ""
          state.profileActiveByDefaultOpt = None
          state.profileDependencies.clear()
          state.profileDependencyManagement.clear()
          state.profileProperties = Map.empty
          state.profileActivationProperties.clear()
          state.profileActivationOsArchOpt = None
          state.profileActivationOsFamilyOpt = None
          state.profileActivationOsNameOpt = None
          state.profileActivationOsVersionOpt = None
          state.profileActivationJdkOpt = None
        }
        def end(state: State) = {
          val p = Profile(
            state.profileId,
            state.profileActiveByDefaultOpt,
            Activation(
              state.profileActivationProperties.toList,
              Activation.Os(
                state.profileActivationOsArchOpt,
                state.profileActivationOsFamilyOpt.toSet,
                state.profileActivationOsNameOpt,
                state.profileActivationOsVersionOpt
              ),
              state.profileActivationJdkOpt
            ),
            state.profileDependencies.toList,
            state.profileDependencyManagement.toList,
            state.profileProperties
          )
          add(state, p)
        }
      },
      content("id" :: prefix) {
        (state, content) =>
          state.profileId = content
      },
      content("activeByDefault" :: "activation" :: prefix) {
        (state, content) =>
          state.profileActiveByDefaultOpt = content match {
            case "true"  => Some(true)
            case "false" => Some(false)
            case _       => None
          }
      },
      new SectionHandler("property" :: "activation" :: prefix) {
        def start(state: State) = {
          state.profilePropertyNameOpt = None
          state.profilePropertyValueOpt = None
        }
        def end(state: State) =
          state.profileActivationProperties +=
            state.profilePropertyNameOpt.get -> state.profilePropertyValueOpt
      },
      content("name" :: "property" :: "activation" :: prefix) {
        (state, content) =>
          state.profilePropertyNameOpt = Some(content)
      },
      content("value" :: "property" :: "activation" :: prefix) {
        (state, content) =>
          state.profilePropertyValueOpt = Some(content)
      },
      content("arch" :: "os" :: "activation" :: prefix) {
        (state, content) =>
          state.profileActivationOsArchOpt = Some(content)
      },
      content("family" :: "os" :: "activation" :: prefix) {
        (state, content) =>
          state.profileActivationOsFamilyOpt = Some(content)
      },
      content("name" :: "os" :: "activation" :: prefix) {
        (state, content) =>
          state.profileActivationOsNameOpt = Some(content)
      },
      content("version" :: "os" :: "activation" :: prefix) {
        (state, content) =>
          state.profileActivationOsVersionOpt = Some(content)
      },
      content("jdk" :: "activation" :: prefix) {
        (state, content) =>
          val s = content
          state.profileActivationJdkOpt =
            Parse.versionInterval(s)
              .orElse(Parse.multiVersionInterval(s))
              .map(Left(_))
              .orElse(Parse.version(s).map(v => Right(Seq(v))))
      }
    ) ++ dependencyHandlers(
      "dependency" :: "dependencies" :: prefix,
      (s, c, d) =>
        s.profileDependencies += c -> d
    ) ++ dependencyHandlers(
      "dependency" :: "dependencies" :: "dependencyManagement" :: prefix,
      (s, c, d) =>
        s.profileDependencyManagement += c -> d
    ) ++ propertyHandlers(
      "properties" :: prefix,
      (s, k, v) =>
        s.profileProperties = s.profileProperties + (k -> v)
    )

  private def dependencyHandlers(
    prefix: List[String],
    add: (State, Configuration, Dependency) => Unit
  ) =
    Seq(
      new SectionHandler(prefix) {
        def start(state: State) = {
          state.dependencyGroupIdOpt = None
          state.dependencyArtifactIdOpt = None
          state.dependencyVersion = ""
          state.dependencyOptional = false
          state.dependencyScope = Configuration.empty
          state.dependencyType = Type.empty
          state.dependencyClassifier = Classifier.empty
          state.dependencyExclusions = Set()
        }
        def end(state: State) = {
          val d = Dependency(
            Module(state.dependencyGroupIdOpt.get, state.dependencyArtifactIdOpt.get, Map.empty),
            state.dependencyVersion,
            Configuration.empty,
            state.dependencyExclusions,
            Attributes(state.dependencyType, state.dependencyClassifier),
            state.dependencyOptional,
            transitive = true
          )
          add(state, state.dependencyScope, d)
        }
      },
      content("groupId" :: prefix) {
        (state, content) =>
          state.dependencyGroupIdOpt = Some(Organization(content))
      },
      content("artifactId" :: prefix) {
        (state, content) =>
          state.dependencyArtifactIdOpt = Some(ModuleName(content))
      },
      content("version" :: prefix) {
        (state, content) =>
          state.dependencyVersion = content
      },
      content("optional" :: prefix) {
        (state, content) =>
          state.dependencyOptional = content == "true"
      },
      content("scope" :: prefix) {
        (state, content) =>
          state.dependencyScope = Configuration(content)
      },
      content("type" :: prefix) {
        (state, content) =>
          state.dependencyType = Type(content)
      },
      content("classifier" :: prefix) {
        (state, content) =>
          state.dependencyClassifier = Classifier(content)
      },
      new SectionHandler("exclusion" :: "exclusions" :: prefix) {
        def start(state: State) = {
          state.dependencyExclusionGroupId = Organization("*")
          state.dependencyExclusionArtifactId = ModuleName("*")
        }
        def end(state: State) = {
          val r = (state.dependencyExclusionGroupId, state.dependencyExclusionArtifactId)
          state.dependencyExclusions = state.dependencyExclusions + r
        }
      },
      content("groupId" :: "exclusion" :: "exclusions" :: prefix) {
        (state, content) =>
          state.dependencyExclusionGroupId = Organization(content)
      },
      content("artifactId" :: "exclusion" :: "exclusions" :: prefix) {
        (state, content) =>
          state.dependencyExclusionArtifactId = ModuleName(content)
      }
    )

  private def propertyHandlers(prefix: List[String], add: (State, String, String) => Unit) =
    Seq(
      new PropertyHandler(prefix) {
        override def name(state: State, name: String) =
          state.propertyNameOpt = Some(name)
        override def content(state: State, content: String) = {
          add(state, state.propertyNameOpt.get, content)
          state.propertyNameOpt = None
        }
      }
    )

  private val handlerMap = handlers
    .map { h =>
      CustomList(h.path) -> h
    }
    .toMap

  private def scmHandlers(prefix: List[String], add: (State, Info.Scm) => Unit) =
    Seq(
      new SectionHandler(prefix) {
        def start(state: State) = {}
        def end(state: State) = {
          val d = Info.Scm(
            url = state.scmUrl,
            connection = state.scmConnection,
            developerConnection = state.scmDeveloperConnection
          )
          add(state, d)
        }
      },
      content("url" :: prefix) {
        (state, content) =>
          state.scmUrl = Some(content)
      },
      content("connection" :: prefix) {
        (state, content) =>
          state.scmConnection = Some(content)
      },
      content("developerConnection" :: prefix) {
        (state, content) =>
          state.scmDeveloperConnection = Some(content)
      }
    )

  private def licenseHandlers(prefix: List[String], add: (State, Info.License) => Unit) =
    Seq(
      new SectionHandler(prefix) {
        def start(state: State): Unit = {}
        def end(state: State): Unit = {
          val license = Info.License(
            state.licenseName,
            state.licenseUrl,
            state.licenseDistribution,
            state.licenseComments
          )
          add(state, license)
        }
      },
      content("name" :: prefix) {
        (state, name) =>
          state.licenseName = name
      },
      content("url" :: prefix) {
        (state, url) =>
          state.licenseUrl = Some(url)
      },
      content("distribution" :: prefix) {
        (state, distribution) =>
          state.licenseDistribution = Some(distribution)
      },
      content("comments" :: prefix) {
        (state, comments) =>
          state.licenseComments = Some(comments)
      }
    )
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy