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

akka.io.dns.DnsResourceRecords.scala Maven / Gradle / Ivy

/*
 * Copyright (C) 2018-2020 Lightbend Inc. 
 */

package akka.io.dns

import java.net.{ Inet4Address, Inet6Address, InetAddress }

import scala.annotation.switch
import scala.concurrent.duration._

import CachePolicy._

import akka.actor.NoSerializationVerificationNeeded
import akka.annotation.DoNotInherit
import akka.annotation.InternalApi
import akka.io.dns.internal.{ DomainName, _ }
import akka.util.{ unused, ByteIterator, ByteString }

/**
 * Not for user extension
 */
@DoNotInherit
sealed abstract class ResourceRecord(
    val name: String,
    val ttl: CachePolicy.Ttl,
    val recType: Short,
    val recClass: Short)
    extends NoSerializationVerificationNeeded {}

final case class ARecord(override val name: String, override val ttl: CachePolicy.Ttl, ip: InetAddress)
    extends ResourceRecord(name, ttl, RecordType.A.code, RecordClass.IN.code) {}

/**
 * INTERNAL API
 */
@InternalApi
private[io] object ARecord {
  def parseBody(name: String, ttl: Ttl, @unused length: Short, it: ByteIterator): ARecord = {
    val address = Array.ofDim[Byte](4)
    it.getBytes(address)
    ARecord(name, ttl, InetAddress.getByAddress(address).asInstanceOf[Inet4Address])
  }
}

final case class AAAARecord(override val name: String, override val ttl: CachePolicy.Ttl, ip: Inet6Address)
    extends ResourceRecord(name, ttl, RecordType.AAAA.code, RecordClass.IN.code) {}

/**
 * INTERNAL API
 */
@InternalApi
private[io] object AAAARecord {

  /**
   * INTERNAL API
   */
  @InternalApi
  def parseBody(name: String, ttl: Ttl, @unused length: Short, it: ByteIterator): AAAARecord = {
    val address = Array.ofDim[Byte](16)
    it.getBytes(address)
    AAAARecord(name, ttl, InetAddress.getByAddress(address).asInstanceOf[Inet6Address])
  }
}

final case class CNameRecord(override val name: String, override val ttl: Ttl, canonicalName: String)
    extends ResourceRecord(name, ttl, RecordType.CNAME.code, RecordClass.IN.code) {}

@InternalApi
private[dns] object CNameRecord {

  /**
   * INTERNAL API
   */
  @InternalApi
  def parseBody(name: String, ttl: Ttl, @unused length: Short, it: ByteIterator, msg: ByteString): CNameRecord = {
    CNameRecord(name, ttl, DomainName.parse(it, msg))
  }
}

final case class SRVRecord(
    override val name: String,
    override val ttl: Ttl,
    priority: Int,
    weight: Int,
    port: Int,
    target: String)
    extends ResourceRecord(name, ttl, RecordType.SRV.code, RecordClass.IN.code) {}

/**
 * INTERNAL API
 */
@InternalApi
private[dns] object SRVRecord {

  /**
   * INTERNAL API
   */
  @InternalApi
  def parseBody(name: String, ttl: Ttl, @unused length: Short, it: ByteIterator, msg: ByteString): SRVRecord = {
    val priority = it.getShort.toInt & 0xFFFF
    val weight = it.getShort.toInt & 0xFFFF
    val port = it.getShort.toInt & 0xFFFF
    SRVRecord(name, ttl, priority, weight, port, DomainName.parse(it, msg))
  }
}

final case class UnknownRecord(
    override val name: String,
    override val ttl: Ttl,
    override val recType: Short,
    override val recClass: Short,
    data: ByteString)
    extends ResourceRecord(name, ttl, recType, recClass) {}

/**
 * INTERNAL API
 */
@InternalApi
private[dns] object UnknownRecord {

  /**
   * INTERNAL API
   */
  @InternalApi
  def parseBody(
      name: String,
      ttl: Ttl,
      recType: Short,
      recClass: Short,
      @unused length: Short,
      it: ByteIterator): UnknownRecord =
    UnknownRecord(name, ttl, recType, recClass, it.toByteString)
}

/**
 * INTERNAL API
 */
@InternalApi
private[dns] object ResourceRecord {

  /**
   * INTERNAL API
   */
  @InternalApi
  def parse(it: ByteIterator, msg: ByteString): ResourceRecord = {
    val name = DomainName.parse(it, msg)
    val recType = it.getShort
    val recClass = it.getShort
    // According to https://www.ietf.org/rfc/rfc1035.txt: "TTL: positive values of a signed 32 bit number."
    val ttl = (it.getInt: @switch) match {
      case 0       => Ttl.never
      case nonZero => Ttl.fromPositive(nonZero.seconds)
    }
    val rdLength = it.getShort
    val data = it.clone().take(rdLength)
    it.drop(rdLength)
    (recType: @switch) match {
      case 1  => ARecord.parseBody(name, ttl, rdLength, data)
      case 5  => CNameRecord.parseBody(name, ttl, rdLength, data, msg)
      case 28 => AAAARecord.parseBody(name, ttl, rdLength, data)
      case 33 => SRVRecord.parseBody(name, ttl, rdLength, data, msg)
      case _  => UnknownRecord.parseBody(name, ttl, recType, recClass, rdLength, data)
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy