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

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

package coursier.ivy

import coursier.util.Traverse.TraverseOps
import coursier.util.ValidationNel
import dataclass.data
import fastparse._, NoWhitespace._

@data class PropertiesPattern(chunks: Seq[PropertiesPattern.ChunkOrProperty]) {

  def string: String = chunks.map(_.string).mkString

  import PropertiesPattern.ChunkOrProperty

  def substituteProperties(properties: Map[String, String]): Either[String, Pattern] = {

    val validation = chunks.validationNelTraverse[String, Seq[Pattern.Chunk]] {
      case prop: ChunkOrProperty.Prop => // (name, alternativesOpt) =>
        properties.get(prop.name) match {
          case Some(value) =>
            ValidationNel.success(Seq(Pattern.Chunk.Const(value)))
          case None =>
            prop.alternative match {
              case Some(alt) =>
                ValidationNel.fromEither(
                  PropertiesPattern(alt)
                    .substituteProperties(properties)
                    .map(_.chunks.toVector)
                )
              case None =>
                ValidationNel.failure(prop.name)
            }
        }

      case opt: ChunkOrProperty.Opt =>
        ValidationNel.fromEither(
          PropertiesPattern(opt.content)
            .substituteProperties(properties)
            .map(l => Seq(Pattern.Chunk.Opt(l.chunks)))
        )

      case v: ChunkOrProperty.Var =>
        ValidationNel.success(Seq(Pattern.Chunk.Var(v.name)))

      case c: ChunkOrProperty.Const =>
        ValidationNel.success(Seq(Pattern.Chunk.Const(c.value)))

    }.map(c => Pattern(c.flatten))

    validation.either.left.map { notFoundProps =>
      s"Property(ies) not found: ${notFoundProps.mkString(", ")}"
    }
  }
}

@data class Pattern(chunks: Seq[Pattern.Chunk]) {

  def +:(chunk: Pattern.Chunk): Pattern =
    Pattern(chunk +: chunks)

  import Pattern.Chunk

  def string: String = chunks.map(_.string).mkString

  def substituteVariables(variables: Map[String, String]): Either[String, String] = {

    def helper(chunks: Seq[Chunk]): ValidationNel[String, Seq[Chunk.Const]] =
      chunks.validationNelTraverse[String, Seq[Chunk.Const]] {
        case v: Chunk.Var =>
          variables.get(v.name) match {
            case Some(value) =>
              ValidationNel.success(Seq(Chunk.Const(value)))
            case None =>
              ValidationNel.failure(v.name)
          }
        case opt: Chunk.Opt =>
          val res = helper(opt.content)
          if (res.isSuccess)
            res
          else
            ValidationNel.success(Seq())
        case c: Chunk.Const =>
          ValidationNel.success(Seq(c))
      }.map(_.flatten)

    val validation = helper(chunks)

    validation.either match {
      case Left(notFoundVariables) =>
        Left(s"Variables not found: ${notFoundVariables.mkString(", ")}")
      case Right(constants) =>
        val b = new StringBuilder
        constants.foreach(b ++= _.value)
        Right(b.result())
    }
  }

  def substitute(varName: String, replacement: Seq[Chunk]): Pattern =
    Pattern(
      chunks.flatMap {
        case v: Chunk.Var if v.name == varName => replacement
        case opt: Chunk.Opt =>
          Seq(Chunk.Opt(Pattern(opt.content).substitute(varName, replacement).chunks))
        case c => Seq(c)
      }
    )

  def substituteDefault: Pattern =
    substitute("defaultPattern", Pattern.default.chunks)

  private lazy val constStart = chunks
    .iterator
    .map {
      case c: Pattern.Chunk.Const => Some(c.value)
      case _                      => None
    }
    .takeWhile(_.nonEmpty)
    .flatten
    .mkString

  def startsWith(s: String): Boolean =
    constStart.startsWith(s)
  def stripPrefix(s: String): Pattern =
    Pattern(
      Pattern.Chunk.Const(constStart.stripPrefix(s)) +:
        chunks.dropWhile { case _: Pattern.Chunk.Const => true; case _ => false }
    )
  def addPrefix(s: String): Pattern =
    Pattern(Pattern.Chunk.Const(s) +: chunks)
}

object PropertiesPattern {

  sealed abstract class ChunkOrProperty extends Product with Serializable {
    def string: String
  }

  object ChunkOrProperty {
    @data class Prop(name: String, alternative: Option[Seq[ChunkOrProperty]])
        extends ChunkOrProperty {
      def string: String =
        s"$${" + name + alternative.fold("")(alt => "-" + alt.map(_.string).mkString) + "}"
    }
    @data class Var(name: String) extends ChunkOrProperty {
      def string: String = "[" + name + "]"
    }
    @data class Opt(content: Seq[ChunkOrProperty]) extends ChunkOrProperty {
      def string: String = "(" + content.map(_.string).mkString + ")"
    }
    object Opt {
      def apply(elem: ChunkOrProperty): Opt =
        Opt(Seq(elem))
      def apply(elem: ChunkOrProperty, elem1: ChunkOrProperty): Opt =
        Opt(Seq(elem, elem1))
      def apply(elem: ChunkOrProperty, elem1: ChunkOrProperty, elem2: ChunkOrProperty): Opt =
        Opt(Seq(elem, elem1, elem2))
    }
    @data class Const(value: String) extends ChunkOrProperty {
      def string: String = value
    }

    implicit def fromString(s: String): ChunkOrProperty = Const(s)
  }

  private def parser[X: P]: P[Seq[ChunkOrProperty]] = {

    val notIn         = s"[]{}()$$".toSet
    def chars         = P(CharsWhile(c => !notIn(c)).!)
    def noHyphenChars = P(CharsWhile(c => !notIn(c) && c != '-').!)

    def constant: P[ChunkOrProperty.Const] =
      P(chars.!)
        .map(ChunkOrProperty.Const(_))

    def property: P[ChunkOrProperty.Prop] =
      P(s"$${" ~ noHyphenChars.! ~ ("-" ~ chunks).? ~ "}")
        .map { case (name, altOpt) => ChunkOrProperty.Prop(name, altOpt) }

    def variable: P[ChunkOrProperty.Var] =
      P("[" ~ chars.! ~ "]")
        .map(ChunkOrProperty.Var(_))

    def optional: P[ChunkOrProperty.Opt] =
      P("(" ~ chunks ~ ")")
        .map(l => ChunkOrProperty.Opt(l))

    def chunks: P[Seq[ChunkOrProperty]] =
      P((constant | property | variable | optional).rep)
        .map(_.toVector) // "Vector" is more readable than "ArrayBuffer"

    chunks
  }

  def parse(pattern: String): Either[String, PropertiesPattern] =
    fastparse.parse(pattern, parser(_)) match {
      case f: Parsed.Failure =>
        Left(f.msg)
      case Parsed.Success(v, _) =>
        Right(PropertiesPattern(v))
    }

}

object Pattern {

  sealed abstract class Chunk extends Product with Serializable {
    def string: String
  }

  object Chunk {
    @data class Var(name: String) extends Chunk {
      def string: String = "[" + name + "]"
    }
    @data class Opt(content: Seq[Chunk]) extends Chunk {
      def string: String = "(" + content.map(_.string).mkString + ")"
    }
    object Opt {
      def apply(chunk: Chunk): Opt =
        Opt(Seq(chunk))
      def apply(chunk: Chunk, chunk1: Chunk): Opt =
        Opt(Seq(chunk, chunk1))
      def apply(chunk: Chunk, chunk1: Chunk, chunk2: Chunk): Opt =
        Opt(Seq(chunk, chunk1, chunk2))
    }
    @data class Const(value: String) extends Chunk {
      def string: String = value
    }

    implicit def fromString(s: String): Chunk = Const(s)
  }

  import Chunk.{Var, Opt}

  // Corresponds to
  //   [organisation]/[module]/(scala_[scalaVersion]/)(sbt_[sbtVersion]/)[revision]/[type]s/[artifact](-[classifier]).[ext]

  val default = Pattern(
    Seq(
      Var("organisation"),
      "/",
      Var("module"),
      "/",
      Opt("scala_", Var("scalaVersion"), "/"),
      Opt("sbt_", Var("sbtVersion"), "/"),
      Var("revision"),
      "/",
      Var("type"),
      "s/",
      Var("artifact"),
      Opt("-", Var("classifier")),
      ".",
      Var("ext")
    )
  )

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy