/* * Copyright 2013-2015 the original author or authors. * * Licensed 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.springframework.xd.dirt.rest; import java.util.ArrayList; import java.util.Iterator; import java.util.List; import java.util.Map; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.domain.Page; import org.springframework.data.domain.Pageable; import org.springframework.data.web.PagedResourcesAssembler; import org.springframework.hateoas.PagedResources; import org.springframework.hateoas.ResourceSupport; import org.springframework.hateoas.mvc.ResourceAssemblerSupport; import org.springframework.http.HttpStatus; import org.springframework.util.StringUtils; import org.springframework.web.bind.annotation.PathVariable; 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.bind.annotation.ResponseStatus; import org.springframework.xd.dirt.core.BaseDefinition; import org.springframework.xd.dirt.core.DeploymentUnitStatus; import org.springframework.xd.dirt.integration.bus.rabbit.NothingToDeleteException; import org.springframework.xd.dirt.integration.bus.rabbit.RabbitBusCleaner; import org.springframework.xd.dirt.server.admin.deployment.DeploymentAction; import org.springframework.xd.dirt.server.admin.deployment.DeploymentMessage; import org.springframework.xd.dirt.server.admin.deployment.DeploymentMessagePublisher; import org.springframework.xd.dirt.server.admin.deployment.DeploymentUnitType; import org.springframework.xd.dirt.stream.AbstractInstancePersistingDeployer; import org.springframework.xd.dirt.stream.BaseInstance; import org.springframework.xd.dirt.stream.DeploymentValidator; import org.springframework.xd.dirt.stream.NoSuchDefinitionException; import org.springframework.xd.rest.domain.DeployableResource; import org.springframework.xd.rest.domain.NamedResource; import org.springframework.xd.rest.domain.support.DeploymentPropertiesFormat; /** * Base Class for XD Controllers. * * @param <D> the kind of definition entity this controller deals with * @param <A> a resource assembler that knows how to build Ts out of Ds * @param <R> the resource class for D * * @author Glenn Renfro * @author Ilayaperumal Gopinathan * @author David Turanski * @author Gunnar Hillert * @author Gary Russell */ public abstract class XDController<D extends BaseDefinition, A extends ResourceAssemblerSupport<D, R>, R extends NamedResource, I extends BaseInstance<D>> { protected final AbstractInstancePersistingDeployer<D, I> deployer; private final DeploymentValidator validator; private A resourceAssemblerSupport; private final RabbitBusCleaner busCleaner = new RabbitBusCleaner(); private final DeploymentUnitType deploymentUnitType; @Autowired private DeploymentMessagePublisher deploymentMessagePublisher; /** * Data holder class for controlling how the list methods should behave. * * @author Eric Bottard */ public static class QueryOptions { public static final QueryOptions NONE = new QueryOptions(); private boolean deployments; /** * Whether to also return deployment status when listing definitions. */ public boolean isDeployments() { return deployments; } public void setDeployments(boolean deployments) { this.deployments = deployments; } } protected XDController(AbstractInstancePersistingDeployer<D, I> deployer, A resourceAssemblerSupport, DeploymentUnitType deploymentUnitType) { this.deployer = deployer; this.validator = deployer; this.resourceAssemblerSupport = resourceAssemblerSupport; this.deploymentUnitType = deploymentUnitType; } /** * Request removal of an existing resource definition (stream or job). * * @param name the name of an existing definition (required) */ @RequestMapping(value = "/definitions/{name}", method = RequestMethod.DELETE) @ResponseStatus(HttpStatus.OK) public void delete(@PathVariable("name") String name) throws Exception { validator.validateBeforeDelete(name); deploymentMessagePublisher.poll(new DeploymentMessage(deploymentUnitType) .setUnitName(name) .setDeploymentAction(DeploymentAction.destroy)); } /** * Request removal of all definitions. */ @RequestMapping(value = "/definitions", method = RequestMethod.DELETE) @ResponseStatus(HttpStatus.OK) public void deleteAll() throws Exception { deploymentMessagePublisher.poll(new DeploymentMessage(deploymentUnitType) .setDeploymentAction(DeploymentAction.destroyAll)); } /** * Request un-deployment of an existing resource. * * @param name the name of an existing resource (required) */ @RequestMapping(value = "/deployments/{name}", method = RequestMethod.DELETE) @ResponseStatus(HttpStatus.OK) public void undeploy(@PathVariable("name") String name) throws Exception { validator.validateBeforeUndeploy(name); deploymentMessagePublisher.poll(new DeploymentMessage(deploymentUnitType) .setUnitName(name) .setDeploymentAction(DeploymentAction.undeploy)); } /** * Request un-deployment of all resources. */ @RequestMapping(value = "/deployments", method = RequestMethod.DELETE) @ResponseStatus(HttpStatus.OK) public void undeployAll() throws Exception { deploymentMessagePublisher.poll(new DeploymentMessage(deploymentUnitType) .setDeploymentAction(DeploymentAction.undeployAll)); } /** * Request deployment of an existing definition resource. The definition must exist before deploying and is included * in the path. A new deployment instance is created. * * @param name the name of an existing definition resource (job or stream) (required) * @param properties the deployment properties for the resource as a comma-delimited list of key=value pairs */ @RequestMapping(value = "/deployments/{name}", method = RequestMethod.POST) @ResponseStatus(HttpStatus.CREATED) public void deploy(@PathVariable("name") String name, @RequestParam(required = false) String properties) throws Exception { Map<String, String> deploymentProperties = DeploymentPropertiesFormat.parseDeploymentProperties(properties); validator.validateBeforeDeploy(name, deploymentProperties); deploymentMessagePublisher.poll(new DeploymentMessage(deploymentUnitType) .setUnitName(name) .setDeploymentAction(DeploymentAction.deploy) .setDeploymentProperties(deploymentProperties)); } /** * Retrieve information about a single {@link ResourceSupport}. * * @param name the name of an existing resource (required) */ @RequestMapping(value = "/definitions/{name}", method = RequestMethod.GET) @ResponseStatus(HttpStatus.OK) @ResponseBody public ResourceSupport display(@PathVariable("name") String name) throws Exception { final D definition = deployer.findOne(name); if (definition == null) { throw new NoSuchDefinitionException(name, "There is no definition named '%s'"); } R resource = resourceAssemblerSupport.toResource(definition); return enhanceWithDeployment(definition, resource); } /** * List module definitions. */ // protected and not annotated with @RequestMapping due to the way // PagedResourcesAssemblerArgumentResolver works // subclasses should override and make public (or delegate) protected PagedResources<R> listValues(Pageable pageable, PagedResourcesAssembler<D> assembler) { Page<D> page = deployer.findAll(pageable); PagedResources<R> result = assembler.toResource(page, resourceAssemblerSupport); if (page.getNumberOfElements() > 0) { enhanceWithDeployments(page, result); } return result; } /** * Queries the deployer about deployed instances and enhances the resources with deployment info. Does nothing if * the operation is not supported. */ private void enhanceWithDeployments(Page<D> page, PagedResources<R> result) { @SuppressWarnings("unchecked") AbstractInstancePersistingDeployer<D, BaseInstance<D>> ipDeployer = (AbstractInstancePersistingDeployer<D, BaseInstance<D>>) deployer; D first = page.getContent().get(0); D last = page.getContent().get(page.getNumberOfElements() - 1); Iterator<BaseInstance<D>> deployedInstances = ipDeployer.deploymentInfo(first.getName(), last.getName()).iterator(); BaseInstance<D> instance = deployedInstances.hasNext() ? deployedInstances.next() : null; // There are >= more definitions than there are instances, and they're both sorted for (R definitionResource : result) { String instanceName = (instance != null) ? instance.getDefinition().getName() : null; if (definitionResource.getName().equals(instanceName)) { ((DeployableResource) definitionResource).setStatus(instance.getStatus().getState().toString()); instance = deployedInstances.hasNext() ? deployedInstances.next() : null; } else { ((DeployableResource) definitionResource).setStatus(DeploymentUnitStatus.State.undeployed.toString()); } } if (deployedInstances.hasNext()) { final List<String> uninspectedInstanceNames = new ArrayList<String>(); while (deployedInstances.hasNext()) { final BaseInstance<D> uninspectedInstance = deployedInstances.next(); uninspectedInstanceNames.add(uninspectedInstance.getDefinition().getName()); } throw new IllegalStateException("Not all instances were looked at: " + StringUtils.collectionToCommaDelimitedString(uninspectedInstanceNames)); } } /** * Create a new resource definition. * * @param name The name of the entity to create (required) * @param definition The entity definition, expressed in the XD DSL (required) */ @RequestMapping(value = "/definitions", method = RequestMethod.POST) @ResponseStatus(HttpStatus.CREATED) public void save(@RequestParam("name") String name, @RequestParam("definition") String definition, @RequestParam(value = "deploy", defaultValue = "true") boolean deploy) throws Exception { DeploymentAction deploymentAction = (deploy) ? DeploymentAction.createAndDeploy : DeploymentAction.create; validator.validateBeforeSave(name, definition); deploymentMessagePublisher.poll(new DeploymentMessage(deploymentUnitType) .setUnitName(name) .setDeploymentAction(deploymentAction) .setDefinition(definition)); } private ResourceSupport enhanceWithDeployment(D definition, R resource) { @SuppressWarnings("unchecked") AbstractInstancePersistingDeployer<D, BaseInstance<D>> ipDeployer = (AbstractInstancePersistingDeployer<D, BaseInstance<D>>) deployer; BaseInstance<D> deployedInstance = ipDeployer.deploymentInfo(definition.getName()); String status = (deployedInstance != null) ? deployedInstance.getStatus().getState().toString() : DeploymentUnitStatus.State.undeployed.toString(); ((DeployableResource) resource).setStatus(status); return resource; } protected abstract D createDefinition(String name, String definition); protected Map<String, List<String>> cleanRabbitBus(String stream, String adminUri, String user, String pw, String vhost, String busPrefix, boolean isJob) { Map<String, List<String>> results = busCleaner.clean(adminUri, user, pw, vhost, busPrefix, stream, isJob); if (results == null || results.size() == 0) { throw new NothingToDeleteException("Nothing to delete for stream " + stream); } return results; } }