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