/** * 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.drill.exec.impersonation.hive; import org.apache.calcite.schema.Schema.TableType; import org.apache.drill.TestBuilder; import org.apache.drill.exec.ExecConstants; import org.apache.drill.exec.dotdrill.DotDrillType; import org.apache.drill.exec.impersonation.BaseTestImpersonation; import org.apache.drill.exec.store.hive.HiveStoragePluginConfig; import org.apache.hadoop.fs.FileSystem; import org.apache.hadoop.fs.Path; import org.apache.hadoop.fs.permission.FsPermission; import org.apache.hadoop.hive.conf.HiveConf; import org.apache.hadoop.hive.conf.HiveConf.ConfVars; import org.apache.hadoop.hive.metastore.MetaStoreUtils; import org.apache.hadoop.hive.shims.ShimLoader; import java.util.List; import java.util.Map; import static org.apache.hadoop.fs.FileSystem.FS_DEFAULT_NAME_KEY; import static org.apache.hadoop.hive.conf.HiveConf.ConfVars.METASTOREURIS; public class BaseTestHiveImpersonation extends BaseTestImpersonation { protected static final String hivePluginName = "hive"; protected static HiveConf hiveConf; protected static String whDir; protected static String studentData; protected static String voterData; protected static final String studentDef = "CREATE TABLE %s.%s" + "(rownum int, name string, age int, gpa float, studentnum bigint) " + "ROW FORMAT DELIMITED FIELDS TERMINATED BY ',' STORED AS TEXTFILE"; protected static final String voterDef = "CREATE TABLE %s.%s" + "(voter_id int,name varchar(30), age tinyint, registration string, " + "contributions double,voterzone smallint,create_time timestamp) " + "ROW FORMAT DELIMITED FIELDS TERMINATED BY ',' STORED AS TEXTFILE"; protected static final String partitionStudentDef = "CREATE TABLE %s.%s" + "(rownum INT, name STRING, gpa FLOAT, studentnum BIGINT) " + "partitioned by (age INT) ROW FORMAT DELIMITED FIELDS TERMINATED BY ',' STORED AS TEXTFILE"; protected static void prepHiveConfAndData() throws Exception { hiveConf = new HiveConf(); // Configure metastore persistence db location on local filesystem final String dbUrl = String.format("jdbc:derby:;databaseName=%s;create=true", getTempDir("metastore_db")); hiveConf.set(ConfVars.METASTORECONNECTURLKEY.varname, dbUrl); hiveConf.set(ConfVars.SCRATCHDIR.varname, "file:///" + getTempDir("scratch_dir")); hiveConf.set(ConfVars.LOCALSCRATCHDIR.varname, getTempDir("local_scratch_dir")); // Set MiniDFS conf in HiveConf hiveConf.set(FS_DEFAULT_NAME_KEY, dfsConf.get(FS_DEFAULT_NAME_KEY)); whDir = hiveConf.get(ConfVars.METASTOREWAREHOUSE.varname); FileSystem.mkdirs(fs, new Path(whDir), new FsPermission((short) 0777)); studentData = getPhysicalFileFromResource("student.txt"); voterData = getPhysicalFileFromResource("voter.txt"); } protected static void startHiveMetaStore() throws Exception { final int port = MetaStoreUtils.findFreePort(); hiveConf.set(METASTOREURIS.varname, "thrift://localhost:" + port); MetaStoreUtils.startMetaStore(port, ShimLoader.getHadoopThriftAuthBridge(), hiveConf); } protected static HiveStoragePluginConfig createHiveStoragePlugin(final Map<String, String> hiveConfig) throws Exception { HiveStoragePluginConfig pluginConfig = new HiveStoragePluginConfig(hiveConfig); pluginConfig.setEnabled(true); return pluginConfig; } protected static Path getWhPathForHiveObject(final String dbName, final String tableName) { if (dbName == null) { return new Path(whDir); } if (tableName == null) { return new Path(whDir, dbName + ".db"); } return new Path(new Path(whDir, dbName + ".db"), tableName); } protected static void addHiveStoragePlugin(final Map<String, String> hiveConfig) throws Exception { getDrillbitContext().getStorage().createOrUpdate(hivePluginName, createHiveStoragePlugin(hiveConfig), true); } protected void showTablesHelper(final String db, List<String> expectedTables) throws Exception { final String dbQualified = hivePluginName + "." + db; final TestBuilder testBuilder = testBuilder() .sqlQuery("SHOW TABLES IN " + dbQualified) .unOrdered() .baselineColumns("TABLE_SCHEMA", "TABLE_NAME"); if (expectedTables.size() == 0) { testBuilder.expectsEmptyResultSet(); } else { for (String tbl : expectedTables) { testBuilder.baselineValues(dbQualified, tbl); } } testBuilder.go(); } protected void fromInfoSchemaHelper(final String pluginName, final String db, List<String> expectedTables, List<TableType> expectedTableTypes) throws Exception { final String dbQualified = pluginName + "." + db; final TestBuilder testBuilder = testBuilder() .sqlQuery("SELECT TABLE_SCHEMA, TABLE_NAME, TABLE_TYPE \n" + "FROM INFORMATION_SCHEMA.`TABLES` \n" + "WHERE TABLE_SCHEMA = '" + dbQualified + "'") .unOrdered() .baselineColumns("TABLE_SCHEMA", "TABLE_NAME", "TABLE_TYPE"); if (expectedTables.size() == 0) { testBuilder.expectsEmptyResultSet(); } else { for (int i = 0; i < expectedTables.size(); ++i) { testBuilder.baselineValues(dbQualified, expectedTables.get(i), expectedTableTypes.get(i).toString()); } } testBuilder.go(); } protected static void createView(final String viewOwner, final String viewGroup, final String viewName, final String viewDef) throws Exception { updateClient(viewOwner); test(String.format("ALTER SESSION SET `%s`='%o';", ExecConstants.NEW_VIEW_DEFAULT_PERMS_KEY, (short) 0750)); test("CREATE VIEW %s.%s.%s AS %s", MINIDFS_STORAGE_PLUGIN_NAME, "tmp", viewName, viewDef); final Path viewFilePath = new Path("/tmp/", viewName + DotDrillType.VIEW.getEnding()); fs.setOwner(viewFilePath, viewOwner, viewGroup); } public static void stopHiveMetaStore() throws Exception { // Unfortunately Hive metastore doesn't provide an API to shut it down. It will be exited as part of the test JVM // exit. As each metastore server instance is using its own resources and not sharing it with other metastore // server instances this should be ok. } }