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

spice.net.URLPath.scala Maven / Gradle / Ivy

package spice.net

import fabric.rw._
import spice.net.URLPathPart._

import java.util.regex.Pattern

case class URLPath(parts: List[URLPathPart]) {
  lazy val absolute: URLPath = {
    var entries = Vector.empty[URLPathPart]
    parts.foreach {
      case UpLevel => entries = entries.dropRight(3)
      case SameLevel => entries = entries.dropRight(1)
      case part => entries = entries :+ part
    }
    val fixed = if (entries.nonEmpty && entries.head != URLPathPart.Separator) {
      URLPathPart.Separator :: entries.toList
    } else {
      entries.toList
    }
    URLPath(fixed)
  }
  lazy val encoded: String = absolute.parts.map(Encoder.apply).mkString
  lazy val decoded: String = absolute.parts.map(part => Decoder(part.value)).mkString
  lazy val asRegexString: String = absolute.parts.map {
    case URLPathPart.Argument(_) => "(.+?)"
    case part => part.value
  }.mkString
  private lazy val pattern = Pattern.compile(asRegexString)

  lazy val arguments: List[String] = parts.collect {
    case part: Argument => part.name
  }

  def withArguments(arguments: Map[String, String]): URLPath = copy(parts = parts.map {
    case part: Argument if arguments.contains(part.name) => Literal(arguments(part.name))
    case part => part
  })

  def withParams(params: (String, String)*): String = if (params.nonEmpty) {
    s"$this?${params.map(t => s"${t._1}=${t._2}").mkString("&")}"
  } else {
    toString
  }

  def extractArguments(literal: URLPath): Map[String, String] = {
    val matcher = pattern.matcher(literal.decoded)
    if (!matcher.matches()) {
      throw new RuntimeException(s"Literal path: ${literal.decoded} was not a match to $asRegexString")
    }
    if (matcher.groupCount() != arguments.length) {
      throw new RuntimeException(s"Arguments ${arguments.length} was not the same as the match groups: ${matcher.groupCount()}")
    }
    arguments.zipWithIndex.map {
      case (name, index) => name -> matcher.group(index + 1)
    }.toMap
  }

  def append(path: String): URLPath = if (path.startsWith("/")) {
    URLPath.parse(path)
  } else {
    val left = parts.dropRight(1)
    val right = URLPath.parse(path, absolutize = false)
    URLPath(left ::: right.parts)
  }

  /**
   * Attempts to match the prefix path against this path and return the remaining path if matched
   */
  def take(prefix: URLPath): Option[URLPath] = {
    val prefixString = prefix.toString
    if (toString.startsWith(prefixString)) {
      Some(URLPath.parse(toString.substring(prefixString.length)))
    } else {
      None
    }
  }

  def merge(that: URLPath): URLPath = URLPath(this.parts ::: that.parts)

  override def equals(obj: Any): Boolean = obj match {
    case that: URLPath => if (this.arguments.nonEmpty == that.arguments.nonEmpty) {
      this.toString == that.toString
    } else if (this.arguments.nonEmpty) {
      val regex = this.asRegexString
      that.decoded.matches(regex)
    } else {
      val regex = that.asRegexString
      this.decoded.matches(regex)
    }
    case _ => false
  }

  override def toString: String = encoded
}

object URLPath {
  implicit val rw: RW[URLPath] = RW.string[URLPath](
    asString = _.encoded,
    fromString = (s: String) => parse(s)
  )

  val empty: URLPath = URLPath(Nil)

  def parse(path: String, absolutize: Boolean = true): URLPath = {
    val parts = path
      .split("((?<=/)|(?=/))")
      .toList
      .flatMap(URLPathPart.apply)

    URLPath(parts) match {
      case p if absolutize => p.absolute
      case p => p
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy