org.apache.kylin.rest.controller.ProjectController Maven / Gradle / Ivy
/*
* 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.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Locale;
import org.apache.commons.lang.StringUtils;
import org.apache.kylin.common.persistence.AclEntity;
import org.apache.kylin.common.util.JsonUtil;
import org.apache.kylin.metadata.project.ProjectInstance;
import org.apache.kylin.rest.exception.BadRequestException;
import org.apache.kylin.rest.exception.ForbiddenException;
import org.apache.kylin.rest.exception.InternalErrorException;
import org.apache.kylin.rest.exception.NotFoundException;
import org.apache.kylin.rest.request.ProjectRequest;
import org.apache.kylin.rest.security.AclPermission;
import org.apache.kylin.rest.service.AccessService;
import org.apache.kylin.rest.service.CubeService;
import org.apache.kylin.rest.service.ProjectService;
import org.apache.kylin.rest.util.AclEvaluate;
import org.apache.kylin.rest.util.ValidateUtil;
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.access.AccessDeniedException;
import org.springframework.security.acls.model.Sid;
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;
/**
* @author xduo
*/
@Controller
@RequestMapping(value = "/projects")
public class ProjectController extends BasicController {
private static final Logger logger = LoggerFactory.getLogger(ProjectController.class);
@Autowired
@Qualifier("projectService")
private ProjectService projectService;
@Autowired
@Qualifier("accessService")
private AccessService accessService;
@Autowired
@Qualifier("cubeMgmtService")
private CubeService cubeService;
@Autowired
@Qualifier("validateUtil")
private ValidateUtil validateUtil;
@Autowired
private AclEvaluate aclEvaluate;
/**
* Get available project list
*
* @return Table metadata array
* @throws IOException
*/
@RequestMapping(value = "", method = { RequestMethod.GET }, produces = { "application/json" })
@ResponseBody
public List getProjects(@RequestParam(value = "limit", required = false) Integer limit,
@RequestParam(value = "offset", required = false) Integer offset) {
return projectService.listProjects(limit, offset);
}
@RequestMapping(value = "/readable", method = { RequestMethod.GET }, produces = { "application/json" })
@ResponseBody
public List getReadableProjects(@RequestParam(value = "limit", required = false) Integer limit,
@RequestParam(value = "offset", required = false) Integer offset) {
List readableProjects = new ArrayList();
//list all projects first
List projectInstances = projectService.listAllProjects(null, null);
for (ProjectInstance projectInstance : projectInstances) {
if (projectInstance == null) {
continue;
}
boolean hasProjectPermission = aclEvaluate.hasProjectReadPermission(projectInstance);
if (hasProjectPermission) {
readableProjects.add(projectInstance);
}
}
int projectLimit = (null == limit) ? Integer.MAX_VALUE : limit;
int projectOffset = (null == offset) ? 0 : offset;
if (readableProjects.size() <= projectOffset) {
return Collections.emptyList();
}
if ((readableProjects.size() - projectOffset) < projectLimit) {
return readableProjects.subList(projectOffset, readableProjects.size());
}
return readableProjects.subList(projectOffset, projectOffset + projectLimit);
}
@RequestMapping(value = "", method = { RequestMethod.POST }, produces = { "application/json" })
@ResponseBody
public ProjectInstance saveProject(@RequestBody ProjectRequest projectRequest) {
ProjectInstance projectDesc = deserializeProjectDesc(projectRequest);
if (StringUtils.isEmpty(projectDesc.getName())) {
throw new InternalErrorException("A project name must be given to create a project");
}
if (!ValidateUtil.isAlphanumericUnderscore(projectDesc.getName())) {
throw new BadRequestException(
String.format(Locale.ROOT,
"Invalid Project name %s, only letters, numbers and underscore supported.",
projectDesc.getName()));
}
ProjectInstance createdProj = null;
try {
createdProj = projectService.createProject(projectDesc);
} catch (Exception e) {
throw new InternalErrorException(e.getLocalizedMessage(), e);
}
return createdProj;
}
@RequestMapping(value = "", method = { RequestMethod.PUT }, produces = { "application/json" })
@ResponseBody
public ProjectInstance updateProject(@RequestBody ProjectRequest projectRequest) {
String formerProjectName = projectRequest.getFormerProjectName();
if (StringUtils.isEmpty(formerProjectName)) {
throw new InternalErrorException("A project name must be given to update a project");
}
ProjectInstance projectDesc = deserializeProjectDesc(projectRequest);
ProjectInstance updatedProj = null;
try {
ProjectInstance currentProject = projectService.getProjectManager().getProject(formerProjectName);
if (currentProject == null) {
throw new NotFoundException("The project named " + formerProjectName + " does not exists");
}
if (projectDesc.getName().equals(currentProject.getName())) {
updatedProj = projectService.updateProject(projectDesc, currentProject);
} else {
throw new IllegalStateException("Rename project is not supported yet, from " + formerProjectName
+ " to " + projectDesc.getName());
}
} catch (Exception e) {
logger.error("Failed to deal with the request.", e);
throw new InternalErrorException(e.getLocalizedMessage(), e);
}
return updatedProj;
}
private ProjectInstance deserializeProjectDesc(ProjectRequest projectRequest) {
ProjectInstance projectDesc = null;
try {
logger.debug("Saving project " + projectRequest.getProjectDescData());
projectDesc = JsonUtil.readValue(projectRequest.getProjectDescData(), ProjectInstance.class);
} catch (Exception e) {
logger.error("Failed to deal with the request.", e);
throw new InternalErrorException("Failed to deal with the request:" + e.getMessage(), e);
}
return projectDesc;
}
@RequestMapping(value = "/{projectName}", method = { RequestMethod.DELETE }, produces = { "application/json" })
@ResponseBody
public void deleteProject(@PathVariable String projectName) {
try {
ProjectInstance project = projectService.getProjectManager().getProject(projectName);
if (project != null) {
projectService.deleteProject(projectName, project);
} else {
logger.info("Project {} not exists", projectName);
}
} catch (Exception e) {
logger.error(e.getLocalizedMessage(), e);
throw new InternalErrorException("Failed to delete project. " + " Caused by: " + e.getMessage(), e);
}
}
@RequestMapping(value = "/{projectName}/owner", method = { RequestMethod.PUT }, produces = {
"application/json" })
@ResponseBody
public ProjectInstance updateProjectOwner(@PathVariable String projectName, @RequestBody String owner) {
ProjectInstance updatedProj;
ProjectInstance currentProject = null;
String oldOwner = null;
boolean updateOwnerSuccess = false;
boolean updateAccessSuccess = false;
try {
validateUtil.checkIdentifiersExists(owner, true);
currentProject = projectService.getProjectManager().getProject(projectName);
if (currentProject == null) {
throw new NotFoundException("The project named " + projectName + " does not exists");
}
oldOwner = currentProject.getOwner();
// update project owner
updatedProj = projectService.updateProjectOwner(currentProject, owner);
updateOwnerSuccess = true;
//grant ADMINISTRATION permission to new owner
AclEntity ae = accessService.getAclEntity("ProjectInstance", currentProject.getUuid());
Sid sid = accessService.getSid(owner, true);
accessService.grant(ae, AclPermission.ADMINISTRATION, sid);
updateAccessSuccess = true;
} catch (AccessDeniedException accessDeniedException) {
throw new ForbiddenException("You don't have right to update this project's owner.");
} catch (Exception e) {
logger.error("Failed to deal with the request.", e);
throw new InternalErrorException(e.getLocalizedMessage(), e);
} finally {
if (!updateAccessSuccess && currentProject != null && updateOwnerSuccess) {
try {
projectService.updateProjectOwner(currentProject, oldOwner);
} catch (IOException e) {
logger.error("Failed to roll back the request.", e);
}
}
}
return updatedProj;
}
public void setProjectService(ProjectService projectService) {
this.projectService = projectService;
}
public void setAccessService(AccessService accessService) {
this.accessService = accessService;
}
public void setCubeService(CubeService cubeService) {
this.cubeService = cubeService;
}
public void setValidateUtil(ValidateUtil validateUtil) {
this.validateUtil = validateUtil;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy