/* * 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.sling.distribution.servlet; import javax.annotation.Nonnull; import javax.servlet.ServletException; import java.io.IOException; import org.apache.felix.scr.annotations.Reference; import org.apache.felix.scr.annotations.sling.SlingServlet; import org.apache.sling.api.SlingHttpServletRequest; import org.apache.sling.api.SlingHttpServletResponse; import org.apache.sling.api.resource.ResourceResolver; import org.apache.sling.api.servlets.SlingAllMethodsServlet; import org.apache.sling.distribution.agent.DistributionAgent; import org.apache.sling.distribution.common.DistributionException; import org.apache.sling.distribution.packaging.DistributionPackage; import org.apache.sling.distribution.packaging.DistributionPackageInfo; import org.apache.sling.distribution.packaging.impl.DistributionPackageUtils; import org.apache.sling.distribution.queue.DistributionQueue; import org.apache.sling.distribution.queue.DistributionQueueEntry; import org.apache.sling.distribution.queue.DistributionQueueItem; import org.apache.sling.distribution.resources.DistributionResourceTypes; import org.apache.sling.distribution.packaging.DistributionPackageBuilder; import org.apache.sling.distribution.packaging.DistributionPackageBuilderProvider; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * Servlet to retrieve a {@link org.apache.sling.distribution.queue.DistributionQueue} status. */ @SuppressWarnings("serial") @SlingServlet(resourceTypes = DistributionResourceTypes.AGENT_QUEUE_RESOURCE_TYPE, methods = {"POST"}) public class DistributionAgentQueueServlet extends SlingAllMethodsServlet { private final Logger log = LoggerFactory.getLogger(getClass()); @Reference private DistributionPackageBuilderProvider packageBuilderProvider; @Override protected void doPost(SlingHttpServletRequest request, SlingHttpServletResponse response) throws ServletException, IOException { String operation = request.getParameter("operation"); DistributionQueue queue = request.getResource().adaptTo(DistributionQueue.class); ResourceResolver resourceResolver = request.getResourceResolver(); if ("delete".equals(operation)) { String limitParam = request.getParameter("limit"); String[] idParam = request.getParameterValues("id"); if (idParam != null) { deleteItems(resourceResolver, queue, idParam); } else { int limit = 1; try { limit = Integer.parseInt(limitParam); } catch (NumberFormatException ex) { log.warn("limit param malformed : "+limitParam, ex); } deleteItems(resourceResolver, queue, limit); } } else if ("copy".equals(operation)) { String from = request.getParameter("from"); String[] idParam = request.getParameterValues("id"); if (idParam != null && from != null) { DistributionAgent agent = request.getResource().getParent().getParent().adaptTo(DistributionAgent.class); DistributionQueue sourceQueue = getQueueOrThrow(agent,from); addItems(resourceResolver, queue, sourceQueue, idParam); } } else if ("move".equals(operation)) { String from = request.getParameter("from"); String[] idParam = request.getParameterValues("id"); if (idParam != null && from != null) { DistributionAgent agent = request.getResource().getParent().getParent().adaptTo(DistributionAgent.class); DistributionQueue sourceQueue = getQueueOrThrow(agent,from); addItems(resourceResolver, queue, sourceQueue, idParam); deleteItems(resourceResolver, sourceQueue, idParam); } } } private void addItems(ResourceResolver resourceResolver, DistributionQueue targetQueue, DistributionQueue sourceQueue, String[] ids) { for (String id: ids) { DistributionQueueEntry entry = sourceQueue.getItem(id); if (entry != null) { targetQueue.add(entry.getItem()); DistributionPackage distributionPackage = getPackage(resourceResolver, entry.getItem()); DistributionPackageUtils.acquire(distributionPackage, targetQueue.getName()); } } } private void deleteItems(ResourceResolver resourceResolver, DistributionQueue queue, int limit) { for (DistributionQueueEntry item : queue.getItems(0, limit)) { deleteItem(resourceResolver, queue, item); } } private void deleteItems(ResourceResolver resourceResolver, DistributionQueue queue, String[] ids) { for (String id : ids) { DistributionQueueEntry entry = queue.getItem(id); deleteItem(resourceResolver, queue, entry); } } private void deleteItem(ResourceResolver resourceResolver, DistributionQueue queue, DistributionQueueEntry entry) { DistributionQueueItem item = entry.getItem(); String id = entry.getId(); queue.remove(id); DistributionPackage distributionPackage = getPackage(resourceResolver, item); DistributionPackageUtils.releaseOrDelete(distributionPackage, queue.getName()); } private DistributionPackage getPackage(ResourceResolver resourceResolver, DistributionQueueItem item) { DistributionPackageInfo info = DistributionPackageUtils.fromQueueItem(item); String type = info.getType(); DistributionPackageBuilder packageBuilder = packageBuilderProvider.getPackageBuilder(type); if (packageBuilder != null) { try { return packageBuilder.getPackage(resourceResolver, item.getPackageId()); } catch (DistributionException e) { log.error("cannot get package", e); } } return null; } @Nonnull private static DistributionQueue getQueueOrThrow(@Nonnull DistributionAgent agent, @Nonnull String queueName) { DistributionQueue queue = agent.getQueue(queueName); if (queue == null) { throw new IllegalArgumentException(String.format("Could not find queue %s", queueName)); } return queue; } }