I'm using spark-redshift (https://github.com/databricks/spark-redshift) which uses avro for transfer.
Reading from Redshift is OK, while writing I'm getting
Caused by: java.lang.NoSuchMethodError: org.apache.avro.generic.GenericData.createDatumWriter(Lorg/apache/avro/Schema;)Lorg/apache/avro/io/DatumWriter
tried using Amazon EMR 4.1.0 (Spark 1.5.0) and 4.0.0 (Spark 1.4.1). Cannot do
import org.apache.avro.generic.GenericData.createDatumWriter
either, just
import org.apache.avro.generic.GenericData
I'm using scala shell Tried download several others avro-mapred and avro jars, tried setting
{"classification":"mapred-site","properties":{"mapreduce.job.user.classpath.first":"true"}},{"classification":"spark-env","properties":{"spark.executor.userClassPathFirst":"true","spark.driver.userClassPathFirst":"true"}}
and adding those jars to spark classpath. Possibly need to tune Hadoop (EMR) somehow.
Does this ring a bell to anyone?
A runntime conflict error in EMR related to Avro is very common. Avro is widely used and a lot of jars have it as a dependancy. I saw few variations of this question with different method in the 'NoSuchMethodError' or different Avro versions.
I failed to solve it with 'spark.executor.userClassPathFirst' flag, because I got LinkageError.
Here is the solution which solved the conflict for me:
When setting the EMR, add a bootstrap action which calls a bash script that download the specific Avro JAR:
#!/bin/bash
When setting the EMR, add the following configuration:
As you can see, I had to add my new library WITH the existing libraries. It didn't work otherwise.
just for reference - workaround by Alex Nastetsky
delete jars from master node
delete jars from slave nodes
Setting configs correctly as proposed by Jonathan is worth a shot too.
spark-redshift
maintainer here.Other EMR users have encountered similar errors when using newer versions of the
spark-avro
library (whichspark-redshift
depends on). In a nutshell, the problem seems to be that EMR's older version of Avro takes precedence over the new version required byspark-avro
. At https://github.com/databricks/spark-avro/issues/91, an issue that seems to match the exception reported here, one user suggested embedding the Avro JARs with their application code: https://github.com/databricks/spark-avro/issues/91#issuecomment-142543149Jonathan from EMR here. Part of the problem is that Hadoop depends upon Avro 1.7.4, and the full Hadoop classpath is included in the Spark path on EMR. It might help for us to upgrade Hadoop's Avro dependency to 1.7.7 so that it matches with Spark's Avro dependency, though I'm a little afraid that this might break something else, but I can try it out anyway.
BTW, one problem I noticed with your example EMR cluster config is that you're using the "spark-env" config classification, whereas the "spark-defaults" classification would be the appropriate one for setting spark.{driver,executor}.userClassPathFirst. I'm not sure this by itself would solve your problem though.