blob: 9d923041a0579f81684d5be623ad8c8c112af719 [file] [log] [blame]
/*
* 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.kylin.rest.controller;
import java.io.IOException;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.kylin.common.util.DateFormat;
import org.apache.kylin.common.util.JsonUtil;
import org.apache.kylin.common.util.StringUtil;
import org.apache.kylin.metadata.model.CsvColumnDesc;
import org.apache.kylin.metadata.model.TableDesc;
import org.apache.kylin.metadata.model.TableExtDesc;
import org.apache.kylin.rest.exception.InternalErrorException;
import org.apache.kylin.rest.exception.NotFoundException;
import org.apache.kylin.rest.request.HiveTableRequest;
import org.apache.kylin.rest.response.TableSnapshotResponse;
import org.apache.kylin.rest.service.TableACLService;
import org.apache.kylin.rest.service.TableService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartFile;
import org.apache.kylin.shaded.com.google.common.collect.Sets;
/**
* @author xduo
*/
@Controller
@RequestMapping(value = "/tables")
public class TableController extends BasicController {
private static final Logger logger = LoggerFactory.getLogger(TableController.class);
@Autowired
@Qualifier("tableService")
private TableService tableService;
@Autowired
@Qualifier("TableAclService")
private TableACLService tableACLService;
/**
* Get available table list of the project
*
* @return Table metadata array
* @throws IOException
*/
@RequestMapping(value = "", method = { RequestMethod.GET }, produces = { "application/json" })
@ResponseBody
public List<TableDesc> getTableDesc(@RequestParam(value = "ext", required = false) boolean withExt,
@RequestParam(value = "project", required = true) String project) throws IOException {
try {
return tableService.getTableDescByProject(project, withExt);
} catch (IOException e) {
logger.error("Failed to get Hive Tables", e);
throw new InternalErrorException(e.getLocalizedMessage(), e);
}
}
// FIXME prj-table
/**
* Get available table list of the input database
*
* @return Table metadata array
* @throws IOException
*/
@RequestMapping(value = "/{project}/{tableName:.+}", method = { RequestMethod.GET }, produces = {
"application/json" })
@ResponseBody
public TableDesc getTableDesc(@PathVariable String tableName, @PathVariable String project) {
TableDesc table = tableService.getTableDescByName(tableName, false, project);
if (table == null)
throw new NotFoundException("Could not find Hive table: " + tableName);
return table;
}
@RequestMapping(value = "/{tables}/{project}", method = { RequestMethod.POST }, produces = { "application/json" })
@ResponseBody
public Map<String, String[]> loadHiveTables(@PathVariable String tables, @PathVariable String project,
@RequestBody HiveTableRequest request) throws IOException {
String submitter = SecurityContextHolder.getContext().getAuthentication().getName();
Map<String, String[]> result = new HashMap<String, String[]>();
String[] tableNames = StringUtil.splitAndTrim(tables, ",");
try {
String[] loaded = tableService.loadHiveTablesToProject(tableNames, project);
result.put("result.loaded", loaded);
Set<String> allTables = new HashSet<String>();
for (String tableName : tableNames) {
allTables.add(tableService.normalizeHiveTableName(tableName));
}
for (String loadedTableName : loaded) {
allTables.remove(loadedTableName);
}
String[] unloaded = new String[allTables.size()];
allTables.toArray(unloaded);
result.put("result.unloaded", unloaded);
// if (request.isCalculate()) {
// tableService.calculateCardinalityIfNotPresent(loaded, submitter, project);
// }
} catch (Throwable e) {
logger.error("Failed to load Hive Table", e);
throw new InternalErrorException(e.getLocalizedMessage(), e);
}
return result;
}
@RequestMapping(value = "/{tables}/{project}", method = { RequestMethod.DELETE }, produces = { "application/json" })
@ResponseBody
public Map<String, String[]> unLoadHiveTables(@PathVariable String tables, @PathVariable String project) {
Set<String> unLoadSuccess = Sets.newHashSet();
Set<String> unLoadFail = Sets.newHashSet();
Map<String, String[]> result = new HashMap<String, String[]>();
try {
for (String tableName : StringUtil.splitByComma(tables)) {
tableACLService.deleteFromTableACLByTbl(project, tableName);
if (tableService.unloadHiveTable(tableName, project)) {
unLoadSuccess.add(tableName);
} else {
unLoadFail.add(tableName);
}
}
} catch (Throwable e) {
logger.error("Failed to unload Hive Table", e);
throw new InternalErrorException(e.getLocalizedMessage(), e);
}
result.put("result.unload.success", (String[]) unLoadSuccess.toArray(new String[unLoadSuccess.size()]));
result.put("result.unload.fail", (String[]) unLoadFail.toArray(new String[unLoadFail.size()]));
return result;
}
// FIXME prj-table
/**
* Regenerate table cardinality
*
* @return Table metadata array
* @throws IOException
*/
// @RequestMapping(value = "/{project}/{tableNames}/cardinality", method = { RequestMethod.PUT }, produces = {
// "application/json" })
// @ResponseBody
// public CardinalityRequest generateCardinality(@PathVariable String tableNames,
// @RequestBody CardinalityRequest request, @PathVariable String project) throws Exception {
// String submitter = SecurityContextHolder.getContext().getAuthentication().getName();
// String[] tables = StringUtil.splitByComma(tableNames);
// try {
// for (String table : tables) {
// tableService.calculateCardinality(table.trim().toUpperCase(Locale.ROOT), submitter, project);
// }
// } catch (IOException e) {
// logger.error("Failed to calculate cardinality", e);
// throw new InternalErrorException(e.getLocalizedMessage(), e);
// }
// return request;
// }
/**
* Show all databases in Hive
*
* @return Hive databases list
* @throws IOException
*/
@RequestMapping(value = "/hive", method = { RequestMethod.GET }, produces = { "application/json" })
@ResponseBody
private List<String> showHiveDatabases(@RequestParam(value = "project", required = false) String project)
throws IOException {
try {
return tableService.getSourceDbNames(project);
} catch (Throwable e) {
logger.error(e.getLocalizedMessage(), e);
throw new InternalErrorException(e.getLocalizedMessage(), e);
}
}
/**
* Show all tables in a Hive database
*
* @return Hive table list
* @throws IOException
*/
@RequestMapping(value = "/hive/{database}", method = { RequestMethod.GET }, produces = { "application/json" })
@ResponseBody
private List<String> showHiveTables(@PathVariable String database,
@RequestParam(value = "project", required = false) String project) throws IOException {
try {
return tableService.getSourceTableNames(project, database);
} catch (Throwable e) {
logger.error(e.getLocalizedMessage(), e);
throw new InternalErrorException(e.getLocalizedMessage(), e);
}
}
@RequestMapping(value = "/{project}/{tableName}/snapshots", method = { RequestMethod.GET })
@ResponseBody
public List<TableSnapshotResponse> getTableSnapshots(@PathVariable final String project,
@PathVariable final String tableName) throws IOException {
throw new UnsupportedOperationException("API getTableSnapshots is not supported in Kylin 4.0 .");
}
@RequestMapping(value = "/supported_datetime_patterns", method = { RequestMethod.GET })
@ResponseBody
public String[] getSupportedDatetimePatterns() {
return DateFormat.SUPPORTED_DATETIME_PATTERN;
}
@RequestMapping(value = "/fetchCsvData", method = { RequestMethod.POST })
@ResponseBody
public List<CsvColumnDesc> fetchCsvData(@RequestParam(value = "file") MultipartFile file,
@RequestParam(value = "withHeader", required = false) boolean withHeader,
@RequestParam(value = "separator", required = false) String separator) throws IOException {
if (file.isEmpty()) {
throw new IllegalArgumentException("Please select a file");
}
return tableService.parseCsvFile(file, withHeader, separator);
}
@RequestMapping(value = "/saveCsvTable", method = { RequestMethod.POST })
@ResponseBody
public TableDesc saveCsvTable(@RequestParam(value = "file") MultipartFile file,
@RequestParam(value = "withHeader", required = false) boolean withHeader,
@RequestParam(value = "separator", required = true) String separator,
@RequestParam(value = "tableName", required = true) String tableName,
@RequestParam(value = "project", required = true) String project,
@RequestParam(value = "columns", required = true) String columnDescList) throws IOException {
if (file.isEmpty()) {
throw new IllegalArgumentException("Please select a file");
}
if (tableService.getTableDescByName(tableName, false, project) != null) {
throw new InternalErrorException("Table " + tableName + " already exists!");
}
TableDesc desc = tableService.generateCsvTableDesc(tableName, JsonUtil.readValue(columnDescList, List.class));
TableExtDesc extDesc = tableService.generateTableExtDesc(desc, withHeader, separator);
tableService.loadTableToProject(desc, extDesc, project);
tableService.saveCsvFile(file, tableName, project);
return desc;
}
public void setTableService(TableService tableService) {
this.tableService = tableService;
}
}