A library for parsing SAS data (sas7bdat) with Spark SQL.
This also includes a SasInputFormat
designed for Hadoop mapreduce. This format is splittable when input is uncompressed
thus can achieve high parallelism for a large SAS file.
This library is inspired by spark-csv and currently uses parso for parsing as it is the only public available parser that handles both forms of SAS compression (CHAR and BINARY).
This library requires Spark 1.4+
This package is published using sbt-spark-package and linking information can be found at http://spark-packages.org/package/saurfang/spark-sas7bdat
This package allows reading SAS files in local or distributed filesystem as Spark DataFrames.
Schema is automatically inferred from meta information embedded in the SAS file.
Thanks to the splittable SasInputFormat
, we are able to convert a 200GB (1.5Bn rows) .sas7bdat file
to .csv files using 2000 executors in under 2 minutes.
SAS data can be queried in pure SQL by registering the data as a (temporary) table.
CREATE TEMPORARY TABLE cars
USING com.github.saurfang.sas.spark
OPTIONS (path "cars.sas7bdat")
The recommended way to load SAS data is using the load functions in SQLContext.
import org.apache.spark.sql.SQLContext
val sqlContext = new SQLContext(sc)
val df = sqlContext.read.format("com.github.saurfang.sas.spark").load("cars.sas7bdat")
df.select("year", "model").write.format("com.databricks.spark.csv").save("newcars.csv") // spark < 2.0.0
df.select("year", "model").write.format("csv").option("header", "true").save("newcars.csv") // spark 2.0.0+
You can also use the implicits from import com.github.saurfang.sas.spark._
.
import org.apache.spark.sql.SQLContext
import com.github.saurfang.sas.spark._
val sqlContext = new SQLContext(sc)
val cars = sqlContext.sasFile("cars.sas7bdat")
// spark < 2.0.0
import com.databricks.spark.csv._
cars.select("year", "model").saveAsCsvFile("newcars.csv")
Similar to the Scala API, SAS data can be loaded using SQLContext.
from pyspark.sql import SQLContext
sqlContext = SQLContext(sc)
df = sqlContext.read.format("com.github.saurfang.sas.spark").load("cars.sas7bdat")
Similar to the Scala API, SAS data can be loaded using SQLContext.
# spark < 1.6.0 (Experimental)
df <- read.df(sqlContext, "cars.sas7bdat", "com.github.saurfang.sas.spark")
# spark 2.0.0+
df <- read.df("cars.sas7bdat", "com.github.saurfang.sas.spark")
We also included a simple SasExport
Spark program that converts .sas7bdat to .csv or .parquet file:
sbt "run input.sas7bdat output.csv"
sbt "run input.sas7bdat output.parquet"
To achieve more parallelism, use spark-submit
script to run it on a Spark cluster. If you don't have a spark
cluster, you can always run it in local mode and take advantage of multi-core.
For further flexibility, you can use spark-shell
:
spark-shell --master local[4] --packages saurfang:spark-sas7bdat:2.0.0-s_2.11
In the shell you can do data analysis like:
import com.github.saurfang.sas.spark._
val random = sqlContext.sasFile("src/test/resources/random.sas7bdat").cache
//random: org.apache.spark.sql.DataFrame = [x: double, f: double]
random.count
//res13: Long = 1000000
random.filter("x > 0.4").count
//res14: Long = 599501
spark-csv
writes outnull
as "null" in csv text output. This means if you read it back for a string type, you might actually read "null" instead ofnull
. The safest option is to export in parquet format where null is properly recorded. See databricks/spark-csv#147 for alternative solution.