blobstore.url.Host.scala Maven / Gradle / Ivy
package blobstore.url
import blobstore.url.exception.{HostParseError, MultipleUrlValidationException}
import blobstore.url.Hostname.Label
import cats.{ApplicativeThrow, ContravariantMonoidal, Order, Show}
import cats.data.{NonEmptyChain, Validated, ValidatedNec}
import cats.data.Validated.{Invalid, Valid}
import cats.syntax.all.*
import scala.util.Try
import scala.util.matching.Regex
/** A Host is any device or computer on a computer network. It can be an IP address or any string that can appear in a
* A, AAAA or CNAME DNS record.
*
* @see
* https://www.ietf.org/rfc/rfc1123.txt 2.1 "Host names and numbers"
*/
sealed trait Host {
def authority: Authority = Authority(this, None, None)
}
object Host {
def parse(s: String): ValidatedNec[HostParseError, Host] = IpV4Address.parse(s).widen[Host].orElse(Hostname.parse(s))
def parseF[F[_]: ApplicativeThrow](s: String) =
parse(s).toEither.leftMap(new MultipleUrlValidationException(_)).liftTo[F]
def unsafe(s: String): Host = parse(s) match {
case Valid(v) => v
case Invalid(e) => throw MultipleUrlValidationException(e) // scalafix:ok
}
implicit def order: Order[Host] =
ContravariantMonoidal[Order].liftContravariant[Host, Either[IpV4Address, Hostname]] {
case a @ IpV4Address(_, _, _, _) => Left(a)
case b @ Hostname(_) => Right(b)
}(Order[Either[IpV4Address, Hostname]])
implicit val ordering: Ordering[Host] = order.toOrdering
implicit val show: Show[Host] = {
case a @ IpV4Address(_, _, _, _) => a.show
case a @ Hostname(_) => a.show
}
}
case class IpV4Address(octet1: Byte, octet2: Byte, octet3: Byte, octet4: Byte) extends Host {
val bytes = (octet1, octet2, octet3, octet4)
override def toString: String = show"${octet1 & 0xff}.${octet2 & 0xff}.${octet3 & 0xff}.${octet4 & 0xff}"
}
object IpV4Address {
def parse(s: String): ValidatedNec[HostParseError, IpV4Address] = {
def predicate(octet: String): ValidatedNec[HostParseError, Byte] =
Try(octet.toInt).toEither.leftMap(_ => HostParseError.ipv4.InvalidIpv4(octet)).flatMap { o =>
if (o >= 0 && o <= 255) o.toByte.asRight[HostParseError]
else HostParseError.ipv4.OctetOutOfRange(o, octet).asLeft[Byte]
}.toValidatedNec
s.split('.').toList match {
case one :: two :: three :: four :: Nil =>
(predicate(one), predicate(two), predicate(three), predicate(four)).mapN(IpV4Address.apply)
case _ => HostParseError.ipv4.InvalidIpv4(s).invalidNec
}
}
def parseF[F[_]: ApplicativeThrow](s: String): F[IpV4Address] =
parse(s).toEither.leftMap(new MultipleUrlValidationException(_)).liftTo[F]
def unsafe(s: String): IpV4Address = parse(s) match {
case Valid(i) => i
case Invalid(e) => throw MultipleUrlValidationException(e) // scalafix:ok
}
private def intOctets(ip: IpV4Address) = (ip.octet1 & 0xff, ip.octet2 & 0xff, ip.octet3 & 0xff, ip.octet4 & 0xff)
def compare(x: IpV4Address, y: IpV4Address): Int =
intOctets(x).compare(intOctets(y))
implicit val ordering: Ordering[IpV4Address] = (x: IpV4Address, y: IpV4Address) => IpV4Address.compare(x, y)
implicit val order: Order[IpV4Address] = Order.fromOrdering[IpV4Address]
implicit val show: Show[IpV4Address] = Show.fromToString[IpV4Address]
}
case class Hostname private (labels: NonEmptyChain[Label]) extends Host {
val value: String = labels.mkString_(".")
override def toString: String = value
}
object Hostname {
/** Label component of a hostname.
*
* Rules
* - 1 to 63 characters
* - Only consists of characters a-z, A-Z, 0-9 or a hyphen
*
* RFC1123, section 2.1 "Host Names and Numbers"
*
* @see
* http://www.ietf.org/rfc/rfc1123.txt
*/
case class Label private[Hostname] (value: String) {
override def toString: String = value
}
object Label {
val Regex: Regex = "[-_a-zA-Z0-9]+".r
implicit val show: Show[Label] = _.value
implicit val order: Order[Label] = (x: Label, y: Label) => x.value.compare(y.value)
implicit val ordering: Ordering[Label] = order.toOrdering
}
def parseF[F[_]: ApplicativeThrow](value: String): F[Hostname] =
parse(value).toEither.leftMap(MultipleUrlValidationException(_)).liftTo[F]
def parse(value: String): ValidatedNec[HostParseError, Hostname] = {
val labels: ValidatedNec[HostParseError, List[Label]] = value.split('.').toList.traverse { el =>
val lengthOk: ValidatedNec[HostParseError, Unit] = Validated.cond(
el.nonEmpty && el.length <= 63,
(),
HostParseError.label.LabelLengthOutOfRange(el)
).toValidatedNec
val charactersOk: ValidatedNec[HostParseError, Unit] =
Validated.cond(
Label.Regex.pattern.matcher(el).matches(),
(),
HostParseError.label.InvalidCharactersInLabel(el)
).toValidatedNec
(lengthOk, charactersOk).tupled.as(Label(el))
}
val labelsNec: ValidatedNec[HostParseError, NonEmptyChain[Label]] = labels.toEither.flatMap {
case h :: t => NonEmptyChain(h, t*).asRight[NonEmptyChain[HostParseError]]
case Nil => NonEmptyChain(HostParseError.hostname.EmptyDomainName).asLeft.leftWiden[NonEmptyChain[HostParseError]]
}.toValidated
val length: ValidatedNec[HostParseError, Unit] =
Validated.cond(
value.length >= 1 && value.length <= 255,
(),
NonEmptyChain(HostParseError.hostname.DomainNameOutOfRange(value))
)
val firstCharacter: ValidatedNec[HostParseError, Unit] = value.headOption.map { h =>
Validated.cond(h.isLetterOrDigit, (), HostParseError.hostname.InvalidFirstCharacter(value)).toValidatedNec[
HostParseError,
Unit
]
}.getOrElse(HostParseError.hostname.EmptyDomainName.invalidNec[Unit])
(labelsNec, length, firstCharacter).mapN((labels, _, _) => Hostname(labels))
}
def unsafe(value: String): Hostname = parse(value) match {
case Valid(h) => h
case Invalid(e) => throw MultipleUrlValidationException(e) // scalafix:ok
}
def compare(a: Hostname, b: Hostname): Int = a.show.compare(b.show)
implicit val ordering: Ordering[Hostname] = (x: Hostname, y: Hostname) => Hostname.compare(x, y)
implicit val order: Order[Hostname] = Order.fromOrdering[Hostname]
implicit val show: Show[Hostname] = _.labels.toList.mkString(".")
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy