All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jppf.client.concurrent.JPPFCompletionService Maven / Gradle / Ivy

There is a newer version: 6.3-alpha
Show newest version
/*
 * JPPF.
 * Copyright (C) 2005-2015 JPPF Team.
 * http://www.jppf.org
 *
 * 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.jppf.client.concurrent;

import java.util.*;
import java.util.concurrent.*;

import org.jppf.client.JPPFJob;
import org.jppf.client.event.*;
import org.jppf.node.protocol.Task;
import org.jppf.utils.LoggingUtils;
import org.slf4j.*;

/**
 * A {@link CompletionService} which works specifically with {@link JPPFExecutorService}s.
 * @param  the type of results returned by the submitted tasks.
 * @author Laurent Cohen
 */
public class JPPFCompletionService implements CompletionService {
  /**
   * Logger for this class.
   */
  private static Logger log = LoggerFactory.getLogger(JPPFCompletionService.class);
  /**
   * Determines whether debug-level logging is enabled.
   */
  private static boolean debugEnabled = LoggingUtils.isDebugEnabled(log);
  /**
   * The executor to which tasks are submitted.
   */
  private final JPPFExecutorService executor;
  /**
   * Maps all the futures for the tasks submitted via this completion service.
   */
  private final Map>> futureMap = new HashMap<>();
  /**
   * Listens to notifications of results received for the tasks submitted via this completion service. 
   */
  private final ResultCollectorListener listener = new ResultCollectorListener();
  /**
   * Holds the queue of futures that can be obtained from this completion service.
   */
  private final BlockingQueue> queue = new LinkedBlockingDeque<>();

  /**
   * Initialize this completion service with the specified executor.
   * @param executor the executor to which tasks are submitted.
   */
  public JPPFCompletionService(final JPPFExecutorService executor) {
    this.executor = executor;
  }

  @Override
  public Future submit(final Callable task) {
    return processFuture((JPPFTaskFuture) executor.submit(task));
  }

  @Override
  public Future submit(final Runnable task, final V result) {
    return processFuture((JPPFTaskFuture) executor.submit(task, result));
  }

  @Override
  public Future take() throws InterruptedException {
    return queue.take();
  }

  @Override
  public Future poll() {
    return queue.poll();
  }

  @Override
  public Future poll(final long timeout, final TimeUnit unit) throws InterruptedException {
    return queue.poll(timeout, unit);
  }

  /**
   * Process the future of a submitted task.
   * @param future the future to process.
   * @return the process future.
   */
  private JPPFTaskFuture processFuture(final JPPFTaskFuture future) {
    JPPFJob job = future.getJob();
    String uuid = future.getJob().getUuid();
    synchronized(futureMap) {
      Map> map = futureMap.get(uuid);
      if (map == null) {
        job.addJobListener(listener);
        map = new HashMap<>();
        futureMap.put(uuid, map);
      }
      map.put(future.getPosition(), future);
    }
    return future;
  }

  /**
   * Process the completion of a task future.
   * @param future the future to process.
   */
  private void processFutureCompletion(final JPPFTaskFuture future) {
    if (future == null) throw new IllegalArgumentException("future should not be null");
    try {
      future.getResult(0L);
    } catch (TimeoutException e) {
      e.printStackTrace();
    }
    queue.offer(future);
  }

  /**
   * Listens to notifications from the FutureResultCollector associated to
   * each job submitted by the {@link JPPFExecutorService}, and updates the queue according
   * to the tasks that are completed.
   */
  private class ResultCollectorListener extends JobListenerAdapter {
    @Override
    public void jobReturned(final JobEvent event) {
      List> tasks = event.getJobTasks();
      if (tasks != null) {
        JPPFJob job = event.getJob();
        String uuid = job.getUuid();
        Map> map = null;
        synchronized(futureMap) {
          map = futureMap.get(uuid);
        }
        if (map == null) return;
        for (Task task: tasks) {
          JPPFTaskFuture future = null;
          synchronized(futureMap) {
            future = map.remove(task.getPosition());
          }
          if (future != null) processFutureCompletion(future);
          if (debugEnabled) log.debug("added future[job uuid=" + uuid + ", position=" + task.getPosition() + "] to the queue");
        }
        synchronized(futureMap) {
          if (map.isEmpty()) futureMap.remove(uuid);
        }
      }
    }

    @Override
    public void jobEnded(final JobEvent event) {
      JPPFJob job = event.getJob();
      String uuid = job.getUuid();
      Map> map = null;
      synchronized(futureMap) {
        map = futureMap.remove(uuid);
      }
      if (map != null) {
        for (Map.Entry> entry: map.entrySet()) {
          JPPFTaskFuture future = entry.getValue();
          processFutureCompletion(future);
          if (debugEnabled) log.debug("added future[job uuid=" + uuid + ", position=" + future.getPosition() + "] to the queue");
        }
        synchronized(futureMap) {
          futureMap.remove(uuid);
        }
      }
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy