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

com.sparkutils.quality.impl.bloom.Parquet.scala Maven / Gradle / Ivy

package com.sparkutils.quality.impl.bloom

import java.io.{ByteArrayInputStream, File, ObjectInputStream}
import com.sparkutils.quality.BloomModel
import com.sparkutils.quality.QualityException.qualityException
import com.sparkutils.quality.impl.bloom.parquet.{BlockSplitBloomFilterImpl, Bloom, BloomFilter, BloomHash, BucketedCreator, BucketedFilesRoot, FileRoot}
import org.apache.spark.sql.catalyst.InternalRow
import org.apache.spark.sql.catalyst.expressions.Expression
import org.apache.spark.sql.catalyst.expressions.aggregate.{ImperativeAggregate, TypedImperativeAggregate}
import org.apache.spark.sql.shim.expressions.InputTypeChecks
import org.apache.spark.sql.types._

trait ParquetBloomAggregator[T <: Bloom[_]] extends InputTypeChecks {
  val child, expectedSizeE, expectedFPPE: Expression
  val mutableAggBufferOffset, inputAggBufferOffset: Int = 0

  // doesn't work with aggregators?
  override def inputDataTypes: Seq[Seq[DataType]] = Seq(BloomHash.hashTypes,
    Seq(LongType, IntegerType, ShortType), Seq(DoubleType))

  protected lazy val expectedSize: Int =
    com.sparkutils.quality.optimalNumOfBits(
      { val eSize = expectedSizeE.eval()
        eSize match {
          case e: Long => e.toInt
          case e: Int => e
          case e: Short => e.toInt
          case _ => qualityException(s"Should have been Short, Long or Integer but was: $eSize ${eSize.getClass.getSimpleName}")
        }},
      {
        val fpp = expectedFPPE.eval()
        fpp match {
          case f: org.apache.spark.sql.types.Decimal => f.toDouble
          case d: Double => d
          case _ => qualityException(s"Should have been Double or Decimal but was: $fpp ${fpp.getClass.getSimpleName}")
        }
      } / 8 )

  lazy val converter = BloomExpressionLookup.bloomLookupValueConverter(child)

  def nullable: Boolean = false

  def dataType: DataType = BinaryType // closest, worse case we lose the input numBuckets, numHashFunctions

  def children: Seq[Expression] = Seq(child, expectedSizeE, expectedFPPE)

}

trait ParquetBloomFPP {
  val expectedSizeE, expectedFPPE: Expression
  protected lazy val fpp = {
    val ef = expectedFPPE.eval()
    ef match {
      case d: Double => d
      case d: org.apache.spark.sql.types.Decimal => d.toDouble
    }
  }
}

trait TypedParquetBloomAggregator[T <: Bloom[_]] extends
  TypedImperativeAggregate[T] with ParquetBloomAggregator[T] {

  implicit def bloomDes: BloomDeserializer[T]

  override def update(buffer: T, input: InternalRow): T = {
    val r = child.eval(input)
    val ar = converter(r)
    buffer += ar
    buffer
  }

  override def merge(buffer: T, input: T): T= {
    (buffer |= input.asInstanceOf[buffer.BloomType]).asInstanceOf[T]
  }

  override def eval(buffer: T): Any = serialize(buffer)

  override def serialize(buffer: T): Array[Byte] = buffer.serialized

  override def deserialize(storageFormat: Array[Byte]): T = Parquet.deserialize[T](storageFormat)

}

case class ParquetAggregator(child: Expression, expectedSizeE: Expression, expectedFPPE: Expression,
                             override val mutableAggBufferOffset: Int = 0,
                             override val inputAggBufferOffset: Int = 0) extends TypedParquetBloomAggregator[BlockSplitBloomFilterImpl] {
  override def withNewMutableAggBufferOffset(newMutableAggBufferOffset: Int): ImperativeAggregate =
  copy(mutableAggBufferOffset = newMutableAggBufferOffset)

  override def withNewInputAggBufferOffset(newInputAggBufferOffset: Int): ImperativeAggregate =
  copy(inputAggBufferOffset = newInputAggBufferOffset)

  override def createAggregationBuffer(): BlockSplitBloomFilterImpl = new BlockSplitBloomFilterImpl(expectedSize)

  implicit def bloomDes: BloomDeserializer[BlockSplitBloomFilterImpl] = Parquet.blockDes

  protected def withNewChildrenInternal(newChildren: IndexedSeq[Expression]): Expression = copy(child = newChildren(0), expectedSizeE = newChildren(1), expectedFPPE = newChildren(2))
}

case class BucketedArrayParquetAggregator(child: Expression, expectedSizeE: Expression, expectedFPPE: Expression, id: Expression,
                                     override val mutableAggBufferOffset: Int = 0,
                                     override val inputAggBufferOffset: Int = 0,
                                     bucketedFilesRoot: BucketedFilesRoot = BucketedFilesRoot(FileRoot(com.sparkutils.quality.bloomFileLocation)))
  extends TypedParquetBloomAggregator[ BucketedCreator[BucketedFilesRoot, Array[Array[Byte]]] ]
  with ParquetBloomFPP {

  protected lazy val numBuckets: Int = {
    val longRes = com.sparkutils.quality.optimalNumberOfBuckets(
      expectedSizeE.dataType match {
        case LongType => expectedSizeE.eval().asInstanceOf[Long]
        case IntegerType => expectedSizeE.eval().asInstanceOf[Int]
        case ShortType => expectedSizeE.eval().asInstanceOf[Short]
      },
      fpp)
    longRes.toByte
  }

  protected lazy val root = bucketedFilesRoot.copy(bloomId = id.eval().toString)

  override def withNewMutableAggBufferOffset(newMutableAggBufferOffset: Int): ImperativeAggregate =
    copy(mutableAggBufferOffset = newMutableAggBufferOffset)

  override def withNewInputAggBufferOffset(newInputAggBufferOffset: Int): ImperativeAggregate =
    copy(inputAggBufferOffset = newInputAggBufferOffset)

  import BucketedCreator.toLocalFiles
  override def createAggregationBuffer(): BucketedCreator[BucketedFilesRoot, Array[Array[Byte]]] = BucketedCreator(expectedSize, numBuckets, fpp, root)

  implicit def bloomDes: BloomDeserializer[BucketedCreator[BucketedFilesRoot, Array[Array[Byte]]]] = Parquet.largeBucketFileDes

  override def eval(buffer: BucketedCreator[BucketedFilesRoot, Array[Array[Byte]]]): Any = serialize(buffer)
  override def dataType: DataType = BinaryType // passes the BroadcastFiles

  override def children: Seq[Expression] = super.children :+ id

  override def inputDataTypes: Seq[Seq[DataType]] = super.inputDataTypes :+ Seq(StringType)
  protected def withNewChildrenInternal(newChildren: IndexedSeq[Expression]): Expression = copy(child = newChildren(0), expectedSizeE = newChildren(1), expectedFPPE = newChildren(2), id = newChildren(3))
}


trait BloomDeserializer[T] {
  def deserialize(storageFormat: Array[Byte]): T
}

// TODO bring this to top level imports
object Parquet {

  implicit val blockDes = new BloomDeserializer[BlockSplitBloomFilterImpl] {
    override def deserialize(storageFormat: Array[Byte]) = BlockSplitBloomFilterImpl(storageFormat)
  }

  implicit val largeBucketFileDes = new BloomDeserializer[BucketedCreator[BucketedFilesRoot, Array[Array[Byte]]]] {
    override def deserialize(storageFormat: Array[Byte]) = {
      val ios = new ByteArrayInputStream(storageFormat)
      val oos = new ObjectInputStream(ios)
      val bucketedFiles = oos.readObject().asInstanceOf[BloomModel]
      oos.close()
      ios.close()
      val ar = bucketedFiles.read

      val bucket = BucketedCreator(ar, bucketedFiles.fpp,
        BucketedFilesRoot(FileRoot(bucketedFiles.rootDir), new File(bucketedFiles.rootDir).getParentFile.getName))
      bucket
    }
  }

  // nb scary looking type lambda is really just turning a two typed into a one typed by fixing the SerializedType
  def deserialize[T: BloomDeserializer](storageFormat: Array[Byte]): T = {
    implicitly[BloomDeserializer[T]].deserialize(storageFormat)
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy