SPARK 3.1.2 Driver端下载UDF jar包导致磁盘爆满

简介: SPARK 3.1.2 Driver端下载UDF jar包导致磁盘爆满

背景

本文基于spark 3.1.2且配置 spark.sql.catalogImplementation=hive

在以spark-sql形式运行sql任务时,发现运行driver端的机器的磁盘总是会达到95%以上的利用率,这样在夜生人静的时候,总会有电话来问候。


分析

经过分析,我们发现是/tmp/${session_id}_resources下的UDF jar包导致的磁盘问题。这就使我们不得怀疑是调用hive的UDF函数造成的,接下来直接说重点,直接到ResolveFunctions Rule,改rule是用来解析函数的规则:

 case u @ UnresolvedFunction(funcId, arguments, isDistinct, filter) =>
            withPosition(u) {
              v1SessionCatalog.lookupFunction(funcId, arguments) match {
                // AggregateWindowFunctions are AggregateFunctions that can only be evaluated within
                // the context of a Window clause. They do not need to be wrapped in an
                // AggregateExpression.
                case wf: AggregateWindowFunction =>
                  if (isDistinct || filter.isDefined) {
                    failAnalysis("DISTINCT or FILTER specified, " +
                      s"but ${wf.prettyName} is not an aggregate function")

这个函数最终会调用SessionCatalog的lookupFunction方法,继而调用loadFunctionResources方法,继而调用HiveSessionResourceLoader的loadResource方法:

class HiveSessionResourceLoader(
    session: SparkSession,
    clientBuilder: () => HiveClient)
  extends SessionResourceLoader(session) {
  private lazy val client = clientBuilder()
  override def addJar(path: String): Unit = {
    val uri = Utils.resolveURI(path)
    resolveJars(uri).foreach { p =>
      client.addJar(p)
      super.addJar(p)
    }
  }
}

之后调用HiveClientImpl.addJar:

class HiveSessionResourceLoader(
    session: SparkSession,
    clientBuilder: () => HiveClient)
  extends SessionResourceLoader(session) {
  private lazy val client = clientBuilder()
  override def addJar(path: String): Unit = {
    val uri = Utils.resolveURI(path)
    resolveJars(uri).foreach { p =>
      client.addJar(p)
      super.addJar(p)
    }
  }
}


注意*runSqlHive(s"ADD JAR $path")*这块代码,这块代码的作用是向hive客户端发动ADD JAR命令,而这个命令的作用就会把对应的UDF JAR包下载到driver端,具体的可参考Hive UDF源码解析【1】Create Function,或者可以跟着代码自己捋清楚(会调用AddResourceProcessor.run方法)。


解决

其实这个问题在spark master分支版本是不存在的,因为有个pr已经间接的解决了这个问题,SPARK-34955.

所以我们的做法很简单,就是直接和并过来对应的commit,事情证明这也很好的解决了这个问题。


说明

其实对于spark来说,下载UDF jar到driver端没有意义的,只有在Task的执行的时候,才会需要对应的UDFjar包,而task所需要的UDFjar是从SessionState的addJar来的

def addJar(path: String): Unit = {
    session.sparkContext.addJar(path)
    val uri = new Path(path).toUri
    val jarURL = if (uri.getScheme == null) {
      // `path` is a local file path without a URL scheme
      new File(path).toURI.toURL
    } else {
      // `path` is a URL with a scheme
      uri.toURL
    }
    session.sharedState.jarClassLoader.addURL(jarURL)
    Thread.currentThread().setContextClassLoader(session.sharedState.jarClassLoader)
  }

session.sparkContext.addJar(path) 方法会把jar包放到driver端,在Task运行的时候,会调用TaskRuner的run()方法:

override def run(): Unit = {
    ...
updateDependencies(
   taskDescription.addedFiles, taskDescription.addedJars, taskDescription.addedArchives)

updateDependencies 方法就会下载task所需要的jar包。


对应的还有SPARK-35286也存在类似问题

相关文章
|
1月前
|
Java Docker 容器
|
1月前
|
运维 Java Shell
Linux非常详细的shell运维脚本一键启动停止状态SpringBoot打成可运行jar包
Linux非常详细的shell运维脚本一键启动停止状态SpringBoot打成可运行jar包
32 0
|
2月前
|
Java Maven 微服务
springboot项目开启远程调试-jar包
springboot项目开启远程调试-jar包
25 0
|
2天前
|
前端开发 Java Linux
宝塔Linux:部署His医疗项目通过jar包的方式
宝塔Linux:部署His医疗项目通过jar包的方式
|
8天前
|
Java
如何解决使用若依前后端分离打包部署到服务器上后主包无法找到从包中的文件的问题?如何在 Java 代码中访问 jar 包中的资源文件?
如何解决使用若依前后端分离打包部署到服务器上后主包无法找到从包中的文件的问题?如何在 Java 代码中访问 jar 包中的资源文件?
43 0
|
10天前
|
Java Maven
springboot jar包启动提示没有主清单属性
springboot jar包启动提示没有主清单属性
|
11天前
|
分布式计算 DataWorks Java
DataWorks产品使用合集之阿里云DataWorks专有云环境下,上传MaxCompute的UDF(用户自定义函数)的JAR包的步骤如何解决
DataWorks作为一站式的数据开发与治理平台,提供了从数据采集、清洗、开发、调度、服务化、质量监控到安全管理的全套解决方案,帮助企业构建高效、规范、安全的大数据处理体系。以下是对DataWorks产品使用合集的概述,涵盖数据处理的各个环节。
21 0
|
21天前
|
Java
JSTL jar包版本错误attribute items does not accept any expressions
确保你在 `items` 属性中使用了一个实际的集合或数组变量,而不是表达式,以解决这个问题。
12 0
|
1月前
|
Java Android开发
读取jar包内外文件
读取jar包内外文件