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

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

package com.sparkutils.quality.impl.bloom

import java.io.{ByteArrayInputStream, ByteArrayOutputStream}

import com.sparkutils.quality.impl.bloom.parquet.BucketedCreator
import com.sparkutils.quality.{BloomLookup, bloomLookup}
import com.sparkutils.quality.impl.rng.RandomLongs
import com.sparkutils.quality.impl.util.{Config, ConfigFactory, Row}
import org.apache.spark.sql._
import org.apache.spark.sql.catalyst.{CatalystTypeConverters, InternalRow}
import org.apache.spark.sql.catalyst.expressions.Expression
import org.apache.spark.sql.functions.expr
import org.apache.spark.util.sketch.BloomFilter

import scala.collection.JavaConverters._

trait BloomFilterTypes {
  /**
   * Used as a param to load the bloomfilter expr
   */
  type BloomFilterMap = BloomExpressionLookup.BloomFilterMap

}

object BloomExpressionLookup {
  /**
   * Used as a param to load the bloomfilter expr
   */
  type BloomFilterMap = Map[ String, (BloomLookup, Double) ]

  /**
   * Identifies a converter for a given expression
   * @param expr
   * @return
   */
  def bloomLookupValueConverter(expr: Expression) =
    if (expr.dataType == RandomLongs.structType)
      (input: Any) => {
        val row = input.asInstanceOf[InternalRow]
        Array(row.getLong(0), row.getLong(1))
      }
    else
      CatalystTypeConverters.createToScalaConverter(expr.dataType)

}

case class BloomRaw(bloom_id: String, bloom: Array[Byte])

case class BloomStruct(fpp: Double, blooms: Array[Array[Byte]])
case class BloomArrayRaw(bloom_id: String, bloom: BloomStruct)

object BloomStruct {
  import frameless._

  implicit val typedEnc = TypedEncoder[BloomStruct]

  implicit val enc = TypedExpressionEncoder[BloomStruct]
}

object BloomArrayRaw {
  import frameless._

  implicit val typedEnc = TypedEncoder[BloomArrayRaw]

  implicit val enc = TypedExpressionEncoder[BloomArrayRaw]
}

trait BloomSerializer[Storage, T] {
  type Serialized = (String, (com.sparkutils.quality.BloomLookup, Double))
  type SerType

  import scala.language.implicitConversions

  implicit def enc(sparkSession: SparkSession): Encoder[SerType]

  def fromType(ser: SerType): Serialized
  def toType(bloomFilter: Serialized): SerType
}

case object SparkBloomFilterSerializer extends BloomSerializer[Array[Byte], com.sparkutils.quality.SparkBloomFilter] {
  type SerType = BloomRaw

  import scala.language.implicitConversions

  implicit def enc(sparkSession: SparkSession): Encoder[SerType] = {
    import sparkSession.implicits._
    implicitly[Encoder[BloomRaw]]
  }

  override def fromType(bloomRaw: BloomRaw): Serialized = {
    val bis = new ByteArrayInputStream(bloomRaw.bloom)

    val bloom = BloomFilter.readFrom(bis)
    bis.close()
    bloomRaw.bloom_id -> (com.sparkutils.quality.SparkBloomFilter(bloom), 1.0 - bloom.expectedFpp())
  }

  override def toType(bloomFilter: Serialized): SerType = {
    val (id, (com.sparkutils.quality.SparkBloomFilter(bloom), fpp)) = bloomFilter

    val bos = new ByteArrayOutputStream()
    bloom.writeTo(bos)
    val bits = bos.toByteArray
    bos.close()
    BloomRaw(id, bits)
  }
}

object Serializing {
  import com.sparkutils.quality.BloomFilterMap

  implicit val sparkBloomFilterSerializer = SparkBloomFilterSerializer

  /**
    * Loads bloomfilters from a dataframe with string id and binary bloomfilter bloom
    * @param id the column representing the bloom filter id, must be string
    * @param bloom the column representing the bloomfilter itself, must be binary
    * @return a map of bloomfilter id to bloomfilter and expected fpp
    */
  def fromDF[SerializedType, T: ({ type B[A] = BloomSerializer[SerializedType, A] })#B](df: DataFrame, id: Column, bloom: Column): BloomFilterMap = {
    val ser = implicitly[BloomSerializer[SerializedType, T]]

    implicit val enc = ser.enc(df.sparkSession)

    df.select(id.as("BLOOM_ID"), bloom.as("BLOOM")).as[ser.SerType].toLocalIterator().asScala.map{
      bloomRaw => ser.fromType(bloomRaw)
    }.toMap
  }

  /**
    * Must have an active sparksession before calling
    * @param bloomFilterMap
    * @return a DataFrame representing storage of ID -> some bloom type
    */
  def toDF[SerializedType, T: ({ type B[A] = BloomSerializer[SerializedType, A] })#B](bloomFilterMap: BloomFilterMap):
      Dataset[BloomSerializer[SerializedType, T]#SerType] = {
    val ser = implicitly[BloomSerializer[SerializedType, T]]

    val blooms = bloomFilterMap.map { pair =>
      ser.toType(pair)
    }.toSeq

    val sess = SparkSession.getDefaultSession.get
    implicit val enc = ser.enc(sess)

    sess.createDataset(blooms).asInstanceOf[Dataset[BloomSerializer[SerializedType, T]#SerType]]
  }

  implicit val factory =
    new ConfigFactory[BloomConfig, BloomRow] {
      override def create(base: Config, row: BloomRow): BloomConfig =
        BloomConfig(base.name, base.source, row.bigBloom, row.value, row.numberOfElements, row.expectedFPP)
    }

  implicit val bloomRowEncoder: Encoder[BloomRow] = Encoders.product[BloomRow]

  def loadBlooms(configs: Seq[BloomConfig]): BloomExpressionLookup.BloomFilterMap =
    configs.map{
      config =>
        import config._

        val df = source.fold(identity, SparkSession.active.sql(_))

        val (lookup, fpp) =
          if (config.bigBloom) {
            val bloom = BucketedCreator.bloomFrom(df, expr(value), numberOfElements, expectedFPP)
            bloom.cleanupOthers()
            (bloomLookup(bloom), 1.0 - bloom.fpp)
          } else {
            val aggrow = df.select(expr(s"smallBloom($value, $numberOfElements, cast( $expectedFPP as double ))")).head()
            val thebytes = aggrow.getAs[Array[Byte]](0)
            (bloomLookup(thebytes), 1.0 - expectedFPP)
          }

        name -> (lookup, fpp)
    }.toMap

}

/**
 * Represents a configuration row for bloom loading
 * @param name the bloom name
 * @param source either a loaded DataFrame or an sql to run against the catalog
 * @param bigBloom when false a small bloom will be attempted
 * @param value the valid expression required to construct the bloom
 */
case class BloomConfig(override val name: String, override val source: Either[DataFrame, String], bigBloom: Boolean, value: String, numberOfElements: Long, expectedFPP: Double) extends Config(name, source)

/**
 * Underlying row information converted into a BloomConfig with the following logic:
 *
 * a) if token is specified sql is ignored
 * b) if token is null sql is used
 * c) if both are null the row will not be used
 */
private[bloom] case class BloomRow(override val name: String, override val token: Option[String],
                                     override val filter: Option[String], override val sql: Option[String], bigBloom: Boolean, value: String, numberOfElements: Long, expectedFPP: Double)
  extends Row(name, token, filter, sql)




© 2015 - 2025 Weber Informatics LLC | Privacy Policy