From 4a99fdeaed31ca9a3b39e29c2f15eb41fb8c3723 Mon Sep 17 00:00:00 2001 From: mpilquist Date: Mon, 5 Feb 2024 09:16:47 -0500 Subject: [PATCH 1/3] Add IpAddress.isPrivate, Ipv4Address.Classes.{A-E}, Ipv4Address.Classes.Private.{A-C}, and Ipv6Address.UniqueLocalBlock --- .../main/scala/com/comcast/ip4s/Host.scala | 47 +++++++++++++++++++ .../com/comcast/ip4s/Ipv4AddressTest.scala | 17 +++++++ .../com/comcast/ip4s/Ipv6AddressTest.scala | 9 ++++ 3 files changed, 73 insertions(+) diff --git a/shared/src/main/scala/com/comcast/ip4s/Host.scala b/shared/src/main/scala/com/comcast/ip4s/Host.scala index 2045fc58..e0680fa8 100644 --- a/shared/src/main/scala/com/comcast/ip4s/Host.scala +++ b/shared/src/main/scala/com/comcast/ip4s/Host.scala @@ -215,6 +215,9 @@ sealed abstract class IpAddress extends IpAddressPlatform with Host with Seriali def asSourceSpecificMulticastLenient: Option[SourceSpecificMulticast[this.type]] = SourceSpecificMulticast.fromIpAddressLenient(this) + /** Returns true if this address is in a private range. */ + def isPrivate: Boolean + /** Narrows this address to an Ipv4Address if that is the underlying type. */ def asIpv4: Option[Ipv4Address] = collapseMappedV4.fold(Some(_), _ => None) @@ -356,6 +359,11 @@ final class Ipv4Address private (protected val bytes: Array[Byte]) extends IpAdd override def isSourceSpecificMulticast: Boolean = this >= Ipv4Address.SourceSpecificMulticastRangeStart && this <= Ipv4Address.SourceSpecificMulticastRangeEnd + override def isPrivate: Boolean = + Ipv4Address.Classes.Private.A.contains(this) || + Ipv4Address.Classes.Private.B.contains(this) || + Ipv4Address.Classes.Private.C.contains(this) + /** Converts this V4 address to a compat V6 address, where the first 12 bytes are all zero and the last 4 bytes * contain the bytes of this V4 address. */ @@ -418,6 +426,38 @@ object Ipv4Address extends Ipv4AddressCompanionPlatform { val SourceSpecificMulticastRangeEnd: Ipv4Address = fromBytes(232, 255, 255, 255) + /** IPv4 address classes represented as CIDRs. */ + object Classes { + + /** Class A: 0.0.0.0 - 127.255.255.255 */ + val A: Cidr[Ipv4Address] = Cidr(fromBytes(0, 0, 0, 0), 1) + + /** Class B: 128.0.0.0 - 191.255.255.255 */ + val B: Cidr[Ipv4Address] = Cidr(fromBytes(128, 0, 0, 0), 2) + + /** Class C: 192.0.0.0 - 223.255.255.255 */ + val C: Cidr[Ipv4Address] = Cidr(fromBytes(192, 0, 0, 0), 3) + + /** Class D: 224.0.0.0 - 239.255.255.255 */ + val D: Cidr[Ipv4Address] = Cidr(fromBytes(224, 0, 0, 0), 4) + + /** Class E: 240.0.0.0 - 255.255.255.255 */ + val E: Cidr[Ipv4Address] = Cidr(fromBytes(240, 0, 0, 0), 5) + + /** Private address ranges. */ + object Private { + + /** Class A: 10.0.0.0 - 10.255.255.255 */ + val A: Cidr[Ipv4Address] = Cidr(fromBytes(10, 0, 0, 0), 8) + + /** Class B: 172.16.0.0 - 172.31.255.255 */ + val B: Cidr[Ipv4Address] = Cidr(fromBytes(172, 16, 0, 0), 12) + + /** Class A: 192.168.0.0 - 192.168.255.255 */ + val C: Cidr[Ipv4Address] = Cidr(fromBytes(192, 168, 0, 0), 16) + } + } + /** Parses an IPv4 address from a dotted-decimal string, returning `None` if the string is not a valid IPv4 address. */ def fromString(value: String): Option[Ipv4Address] = { @@ -609,6 +649,9 @@ final class Ipv6Address private (protected val bytes: Array[Byte]) extends IpAdd override def isSourceSpecificMulticast: Boolean = this >= Ipv6Address.SourceSpecificMulticastRangeStart && this <= Ipv6Address.SourceSpecificMulticastRangeEnd + override def isPrivate: Boolean = + Ipv6Address.UniqueLocalBlock.contains(this) || (isMappedV4 && collapseMappedV4.isPrivate) + /** Applies the supplied mask to this address. * * @example {{{ @@ -653,6 +696,10 @@ object Ipv6Address extends Ipv6AddressCompanionPlatform { val MappedV4Block: Cidr[Ipv6Address] = Cidr(Ipv6Address.fromBytes(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 255, 255, 0, 0, 0, 0), 96) + /** CIDR which defines unique local address block. */ + val UniqueLocalBlock: Cidr[Ipv6Address] = + Cidr(Ipv6Address.fromBytes(0xfc, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0), 7) + /** Parses an IPv6 address from a string in RFC4291 notation, returning `None` if the string is not a valid IPv6 * address. */ diff --git a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala index a955cd6d..a190ff80 100644 --- a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala +++ b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala @@ -67,4 +67,21 @@ class Ipv4AddressTest extends BaseTestSuite { assertEquals(Ipv4Address.fromString("0.0.0.0").map(_.previous), Ipv4Address.fromString("255.255.255.255")) forAll { (ip: Ipv4Address) => assertEquals(ip.previous, Ipv4Address.fromLong(ip.toLong - 1)) } } + + test("isPrivate") { + assert(!ipv4"10.0.0.0".previous.isPrivate) + assert(ipv4"10.0.0.0".isPrivate) + assert(ipv4"10.255.255.255".isPrivate) + assert(!ipv4"10.255.255.255".next.isPrivate) + + assert(!ipv4"172.16.0.0".previous.isPrivate) + assert(ipv4"172.16.0.0".isPrivate) + assert(ipv4"172.31.255.255".isPrivate) + assert(!ipv4"172.31.255.255".next.isPrivate) + + assert(!ipv4"192.168.0.0".previous.isPrivate) + assert(ipv4"192.168.0.0".isPrivate) + assert(ipv4"192.168.255.255".isPrivate) + assert(!ipv4"192.168.255.255".next.isPrivate) + } } diff --git a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala index 4cab33c3..4d845f7d 100644 --- a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala +++ b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala @@ -131,4 +131,13 @@ class Ipv6AddressTest extends BaseTestSuite { assertEquals[Any, Any](addr.asIpv6, Some(addr)) assertEquals[Any, Any](addr.asIpv4, Some(ip"0.15.0.15")) } + + test("isPrivate") { + assert(!ipv6"fc00::".previous.isPrivate) + assert(ipv6"fc00::".isPrivate) + assert(ipv6"fe00::".previous.isPrivate) + assert(!ipv6"fe00::".isPrivate) + // mapped v4 + assert(ipv6"::ffff:10.1.1.1".isPrivate) + } } From 4df99ca5043e1fa281abf07b9805491f8b1b2a6b Mon Sep 17 00:00:00 2001 From: mpilquist Date: Mon, 5 Feb 2024 09:26:53 -0500 Subject: [PATCH 2/3] Update mima exclusions --- build.sbt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/build.sbt b/build.sbt index f046459b..73f3df9f 100644 --- a/build.sbt +++ b/build.sbt @@ -32,7 +32,8 @@ ThisBuild / mimaBinaryIssueFilters ++= Seq( ProblemFilters.exclude[DirectMissingMethodProblem]("com.comcast.ip4s.Multicast.ordinal"), ProblemFilters.exclude[MissingTypesProblem]("com.comcast.ip4s.Multicast$"), ProblemFilters.exclude[DirectMissingMethodProblem]("com.comcast.ip4s.SourceSpecificMulticast.ordinal"), - ProblemFilters.exclude[MissingTypesProblem]("com.comcast.ip4s.SourceSpecificMulticast$") + ProblemFilters.exclude[MissingTypesProblem]("com.comcast.ip4s.SourceSpecificMulticast$"), + ProblemFilters.exclude[ReversedMissingMethodProblem]("com.comcast.ip4s.IpAddress.isPrivate") // #562 ) lazy val root = tlCrossRootProject.aggregate(core, testKit) From 01040d636af4195b3ce03701c4604c9e9c5bcf62 Mon Sep 17 00:00:00 2001 From: mpilquist Date: Mon, 5 Feb 2024 10:16:52 -0500 Subject: [PATCH 3/3] Add isLoopback and isLinkLocal to IpAddress --- build.sbt | 4 ++- .../main/scala/com/comcast/ip4s/Host.scala | 33 ++++++++++++++++++- .../com/comcast/ip4s/Ipv4AddressTest.scala | 13 ++++++++ .../com/comcast/ip4s/Ipv6AddressTest.scala | 10 ++++++ 4 files changed, 58 insertions(+), 2 deletions(-) diff --git a/build.sbt b/build.sbt index 73f3df9f..143aef27 100644 --- a/build.sbt +++ b/build.sbt @@ -33,7 +33,9 @@ ThisBuild / mimaBinaryIssueFilters ++= Seq( ProblemFilters.exclude[MissingTypesProblem]("com.comcast.ip4s.Multicast$"), ProblemFilters.exclude[DirectMissingMethodProblem]("com.comcast.ip4s.SourceSpecificMulticast.ordinal"), ProblemFilters.exclude[MissingTypesProblem]("com.comcast.ip4s.SourceSpecificMulticast$"), - ProblemFilters.exclude[ReversedMissingMethodProblem]("com.comcast.ip4s.IpAddress.isPrivate") // #562 + ProblemFilters.exclude[ReversedMissingMethodProblem]("com.comcast.ip4s.IpAddress.isPrivate"), // #562 + ProblemFilters.exclude[ReversedMissingMethodProblem]("com.comcast.ip4s.IpAddress.isLoopback"), + ProblemFilters.exclude[ReversedMissingMethodProblem]("com.comcast.ip4s.IpAddress.isLinkLocal") ) lazy val root = tlCrossRootProject.aggregate(core, testKit) diff --git a/shared/src/main/scala/com/comcast/ip4s/Host.scala b/shared/src/main/scala/com/comcast/ip4s/Host.scala index e0680fa8..88cd7b1d 100644 --- a/shared/src/main/scala/com/comcast/ip4s/Host.scala +++ b/shared/src/main/scala/com/comcast/ip4s/Host.scala @@ -215,6 +215,12 @@ sealed abstract class IpAddress extends IpAddressPlatform with Host with Seriali def asSourceSpecificMulticastLenient: Option[SourceSpecificMulticast[this.type]] = SourceSpecificMulticast.fromIpAddressLenient(this) + /** Returns true if this address is a loopback address. */ + def isLoopback: Boolean + + /** Returns true if this address is a link local address. */ + def isLinkLocal: Boolean + /** Returns true if this address is in a private range. */ def isPrivate: Boolean @@ -359,6 +365,12 @@ final class Ipv4Address private (protected val bytes: Array[Byte]) extends IpAdd override def isSourceSpecificMulticast: Boolean = this >= Ipv4Address.SourceSpecificMulticastRangeStart && this <= Ipv4Address.SourceSpecificMulticastRangeEnd + override def isLoopback: Boolean = + Ipv4Address.Classes.Loopback.contains(this) + + override def isLinkLocal: Boolean = + Ipv4Address.Classes.LinkLocal.contains(this) + override def isPrivate: Boolean = Ipv4Address.Classes.Private.A.contains(this) || Ipv4Address.Classes.Private.B.contains(this) || @@ -456,6 +468,12 @@ object Ipv4Address extends Ipv4AddressCompanionPlatform { /** Class A: 192.168.0.0 - 192.168.255.255 */ val C: Cidr[Ipv4Address] = Cidr(fromBytes(192, 168, 0, 0), 16) } + + /** Loopback: 127.0.0.0 - 127.255.255.255. */ + val Loopback: Cidr[Ipv4Address] = Cidr(fromBytes(127, 0, 0, 0), 8) + + /** Link local: 169.254.0.0 - 169.254.255.255. */ + val LinkLocal: Cidr[Ipv4Address] = Cidr(fromBytes(169, 254, 0, 0), 16) } /** Parses an IPv4 address from a dotted-decimal string, returning `None` if the string is not a valid IPv4 address. @@ -649,6 +667,12 @@ final class Ipv6Address private (protected val bytes: Array[Byte]) extends IpAdd override def isSourceSpecificMulticast: Boolean = this >= Ipv6Address.SourceSpecificMulticastRangeStart && this <= Ipv6Address.SourceSpecificMulticastRangeEnd + override def isLoopback: Boolean = + this == Ipv6Address.Loopback || (isMappedV4 && collapseMappedV4.isLoopback) + + override def isLinkLocal: Boolean = + Ipv6Address.LinkLocalBlock.contains(this) || (isMappedV4 && collapseMappedV4.isLinkLocal) + override def isPrivate: Boolean = Ipv6Address.UniqueLocalBlock.contains(this) || (isMappedV4 && collapseMappedV4.isPrivate) @@ -696,10 +720,17 @@ object Ipv6Address extends Ipv6AddressCompanionPlatform { val MappedV4Block: Cidr[Ipv6Address] = Cidr(Ipv6Address.fromBytes(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 255, 255, 0, 0, 0, 0), 96) - /** CIDR which defines unique local address block. */ + /** Alias for ::1. */ + val Loopback: Ipv6Address = Ipv6Address.fromBytes(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1) + + /** CIDR which defines the unique local address block. */ val UniqueLocalBlock: Cidr[Ipv6Address] = Cidr(Ipv6Address.fromBytes(0xfc, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0), 7) + /** CIDR which defines the linked scope unicast address block. */ + val LinkLocalBlock: Cidr[Ipv6Address] = + Cidr(Ipv6Address.fromBytes(0xfe, 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0), 10) + /** Parses an IPv6 address from a string in RFC4291 notation, returning `None` if the string is not a valid IPv6 * address. */ diff --git a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala index a190ff80..a360381a 100644 --- a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala +++ b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv4AddressTest.scala @@ -84,4 +84,17 @@ class Ipv4AddressTest extends BaseTestSuite { assert(ipv4"192.168.255.255".isPrivate) assert(!ipv4"192.168.255.255".next.isPrivate) } + + test("isLoopback") { + assert(ipv4"127.0.0.1".isLoopback) + assert(ipv4"127.255.255.255".isLoopback) + assert(!ipv4"128.0.0.0".isLoopback) + } + + test("isLinkLocal") { + assert(!ipv4"127.0.0.1".isLinkLocal) + assert(ipv4"169.254.0.0".isLinkLocal) + assert(ipv4"169.254.255.255".isLinkLocal) + assert(!ipv4"169.254.255.255".next.isLinkLocal) + } } diff --git a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala index 4d845f7d..476786aa 100644 --- a/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala +++ b/test-kit/shared/src/test/scala/com/comcast/ip4s/Ipv6AddressTest.scala @@ -140,4 +140,14 @@ class Ipv6AddressTest extends BaseTestSuite { // mapped v4 assert(ipv6"::ffff:10.1.1.1".isPrivate) } + + test("isLoopback") { + assert(ipv6"::1".isLoopback) + assert(ipv6"::ffff:127.0.0.1".isLoopback) + } + + test("isLinkLocal") { + assert(ipv6"fe80::1".isLinkLocal) + assert(ipv6"::ffff:169.254.0.0".isLinkLocal) + } }