Skip to content
Snippets Groups Projects
Commit 9b0593d5 authored by Ryan Blue's avatar Ryan Blue Committed by Reynold Xin
Browse files

[SPARK-18086] Add support for Hive session vars.

## What changes were proposed in this pull request?

This adds support for Hive variables:

* Makes values set via `spark-sql --hivevar name=value` accessible
* Adds `getHiveVar` and `setHiveVar` to the `HiveClient` interface
* Adds a SessionVariables trait for sessions like Hive that support variables (including Hive vars)
* Adds SessionVariables support to variable substitution
* Adds SessionVariables support to the SET command

## How was this patch tested?

* Adds a test to all supported Hive versions for accessing Hive variables
* Adds HiveVariableSubstitutionSuite

Author: Ryan Blue <blue@apache.org>

Closes #15738 from rdblue/SPARK-18086-add-hivevar-support.
parent 3eda0570
No related branches found
No related tags found
No related merge requests found
......@@ -60,6 +60,13 @@ case class SetCommand(kv: Option[(String, Option[String])]) extends RunnableComm
}
(keyValueOutput, runFunc)
case Some((key @ SetCommand.VariableName(name), Some(value))) =>
val runFunc = (sparkSession: SparkSession) => {
sparkSession.conf.set(name, value)
Seq(Row(key, value))
}
(keyValueOutput, runFunc)
// Configures a single property.
case Some((key, Some(value))) =>
val runFunc = (sparkSession: SparkSession) => {
......@@ -117,6 +124,10 @@ case class SetCommand(kv: Option[(String, Option[String])]) extends RunnableComm
}
object SetCommand {
val VariableName = """hivevar:([^=]+)""".r
}
/**
* This command is for resetting SQLConf to the default values. Command that runs
* {{{
......
......@@ -17,10 +17,7 @@
package org.apache.spark.sql.internal
import java.util.regex.Pattern
import org.apache.spark.internal.config._
import org.apache.spark.sql.AnalysisException
/**
* A helper class that enables substitution using syntax like
......@@ -37,6 +34,7 @@ class VariableSubstitution(conf: SQLConf) {
private val reader = new ConfigReader(provider)
.bind("spark", provider)
.bind("sparkconf", provider)
.bind("hivevar", provider)
.bind("hiveconf", provider)
/**
......@@ -49,5 +47,4 @@ class VariableSubstitution(conf: SQLConf) {
input
}
}
}
......@@ -38,7 +38,7 @@ import org.apache.thrift.transport.TSocket
import org.apache.spark.internal.Logging
import org.apache.spark.sql.AnalysisException
import org.apache.spark.sql.hive.HiveUtils
import org.apache.spark.sql.hive.{HiveSessionState, HiveUtils}
import org.apache.spark.util.ShutdownHookManager
/**
......@@ -291,6 +291,10 @@ private[hive] class SparkSQLCLIDriver extends CliDriver with Logging {
throw new RuntimeException("Remote operations not supported")
}
override def setHiveVariables(hiveVariables: java.util.Map[String, String]): Unit = {
hiveVariables.asScala.foreach(kv => SparkSQLEnv.sqlContext.conf.setConfString(kv._1, kv._2))
}
override def processCmd(cmd: String): Int = {
val cmd_trimmed: String = cmd.trim()
val cmd_lower = cmd_trimmed.toLowerCase(Locale.ENGLISH)
......
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.spark.sql.hive
import org.apache.spark.sql.{QueryTest, Row}
import org.apache.spark.sql.hive.test.TestHiveSingleton
class HiveVariableSubstitutionSuite extends QueryTest with TestHiveSingleton {
test("SET hivevar with prefix") {
spark.sql("SET hivevar:county=gram")
assert(spark.conf.getOption("county") === Some("gram"))
}
test("SET hivevar with dotted name") {
spark.sql("SET hivevar:eloquent.mosquito.alphabet=zip")
assert(spark.conf.getOption("eloquent.mosquito.alphabet") === Some("zip"))
}
test("hivevar substitution") {
spark.conf.set("pond", "bus")
checkAnswer(spark.sql("SELECT '${hivevar:pond}'"), Row("bus") :: Nil)
}
test("variable substitution without a prefix") {
spark.sql("SET hivevar:flask=plaid")
checkAnswer(spark.sql("SELECT '${flask}'"), Row("plaid") :: Nil)
}
test("variable substitution precedence") {
spark.conf.set("turn.aloof", "questionable")
spark.sql("SET hivevar:turn.aloof=dime")
// hivevar clobbers the conf setting
checkAnswer(spark.sql("SELECT '${turn.aloof}'"), Row("dime") :: Nil)
}
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment