When I write out a dataframe to, say, csv, a .csv file is created for each partition. Suppose I want to limit the max size of each file to, say, 1 MB. I could do the write multiple times and increase the argument to repartition each time. Is there a way I can calculate ahead of time what argument to use for repartition to ensure the max size of each file is less than some specified size.
I imagine there might be pathological cases where all the data ends up on one partition. So make the weaker assumption that we only want to ensure that the average file size is less than some specified amount, say 1 MB.
I was trying to find out some clever idea that would not kill the cluster at the same time and the only thing that came to my mind was:
The code should look more like this:
val df: DataFrame = ??? // your df
val rowSize = getBytes(df.head)
val rowCount = df.count()
val partitionSize = 1000000 // million bytes in MB?
val noPartitions: Int = (rowSize * rowCount / partitionSize).toInt
df.repartition(noPartitions).write.format(...) // save to csv
// just helper function from https://stackoverflow.com/a/39371571/1549135
def getBytes(value: Any): Long = {
val stream: ByteArrayOutputStream = new ByteArrayOutputStream()
val oos = new ObjectOutputStream(stream)
oos.writeObject(value)
oos.close
stream.toByteArray.length
}
While my first choice was to calculate each row's byte size, that would be terribly inefficient. So, unless your data size in each row differs in size greatly, I would say that this solution will work. You can also calculate every n-th row size. You got the idea.
Also, I just 'hope' that Long
will be big enough to support the expected size to calculate noPartitions
. If not (if you have a lot of rows), maybe it would be better to change the operations order, f.e.:
val noPartitions: Int = (rowSize / partitionSize * rowCount).toInt
again this is just a drafted idea with no domain knowledge about your data.
While going through the apache-spark docs I have found an interesting cross-system solution:
spark.sql.files.maxPartitionBytes
which sets:
The maximum number of bytes to pack into a single partition when reading files.
The default value is 134217728 (128 MB)
.
So I suppose you could set it to 1000000 (1MB)
and it will have a permanent effect on your DataFrames
. However, too small partition size may greatly impact your performance!
You can set it up, during SparkSession
creation:
val spark = SparkSession
.builder()
.appName("Spark SQL basic example")
.config("spark.sql.files.maxPartitionBytes", 100000)
.getOrCreate()
All of above is only valid if (I remember correctly and) the csv is partitioned with the same number of files as there are partitions of DataFrame.
val df = spark.range(10000000)
df.cache
val catalyst_plan = df.queryExecution.logical
val df_size_in_bytes = spark.sessionState.executePlan(catalyst_plan).optimizedPlan.stats.sizeInBytes
df_size_in_bytes: BigInt = 80000000
The best solution would be take 100 records and estimate the size and apply for all the rows as the above example
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With