src/main/scala/net/tz/ip/IpNetwork.scala
author Tomas Zeman <tzeman@volny.cz>
Fri, 10 Feb 2012 10:55:23 +0100
changeset 29 96e47961c240
parent 25 6ed154cb8b32
permissions -rw-r--r--
tz-lift-misc: 0.3 -> 0.4-SNAPSHOT

/*
 * Copyright 2011 Tomas Zeman <tzeman@volny.cz>
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package net.tz.ip

import net.liftweb.common._
import net.liftweb.util.Helpers._

case class IpNetwork(from: IpAddress, to: IpAddress) {
  require(from.v == to.v, "IpAddresses must be of the same version")

  lazy val v = from.v
  lazy val mask = from.v.ones ^ from.toBigInt ^ to.toBigInt
  lazy val size = to.toBigInt - from.toBigInt + 1
  lazy val maskLen = mask bitCount
  lazy val maskIp = IpAddress(v, mask)

  override def toString = maskIp map { _ => from + "/" + maskLen } openOr
    (from + "-" + to)
}

object IpNetwork {
  def apply(net: String, maskLen: Int): Box[IpNetwork] = for {
    ip <- IpAddress(net)
    n <- apply(ip, maskLen)
  } yield { n }

  def apply(net: IpAddress, maskLen: Int): Box[IpNetwork] = {
    require(net.v.bits >= maskLen,
      "MaskLen must be less or equal to IP length")
    val hm = BigInt(2).pow(net.v.bits - maskLen) - 1 // host mask
    val m = net.v.ones &~ hm // net mask
    val f = net.toBigInt & m
    val t = net.toBigInt | hm
    for {
      from <- IpAddress(net.v, f)
      to <- IpAddress(net.v, t)
    } yield { IpNetwork(from, to) }
  }
}

// vim: set ts=2 sw=2 et: