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

com.dasasian.chok.client.WorkQueue Maven / Gradle / Ivy

There is a newer version: 1.7
Show newest version
/**
 * Copyright (C) 2014 Dasasian ([email protected])
 *
 * 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 com.dasasian.chok.client;

import com.dasasian.chok.client.ClientResult.IClosedListener;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.RejectedExecutionException;

/**
 * This class manages the multiple NodeInteraction threads for a call.
 * The initial node interactions and any resulting retries go through the
 * same execute() method. We allow blocking or non-blocking access
 * to the result set, or you can provide a custom policy to control the
 * length of time spent waiting for results to complete.
 */
class WorkQueue implements INodeExecutor {

    private static final Logger LOG = LoggerFactory.getLogger(WorkQueue.class);

    private static int instanceCounter = 0;
    private final INodeInteractionFactory interactionFactory;
    private final INodeProxyManager shardManager;
    private final Method method;
    private final int shardArrayParamIndex;
    private final Object[] args;
    private final ExecutorService executor = Executors.newCachedThreadPool();
    private final ClientResult results;
    private final int instanceId = instanceCounter++;
    private int callCounter = 0;
    /**
     * Normal constructor. Jobs submitted by execute() will result in a
     * NodeInteraction instance being created and run(). The WorkQueue
     * is initially emtpy. Call execute() to add jobs.
     * 

* DO NOT CHANGE THE ARGUMENTS WHILE THIS CALL IS RUNNING OR YOU WILL BE * SORRY. * * @param shardManager The class that maintains the node/shard maps, the node selection * policy, and the node proxies. * @param allShards The entire set of shards for this request. When all these shards * have reported in, the result is complete. * @param method Which method to call on the server side. * @param shardArrayParamIndex Which paramater, if any, should be overwritten with an array of * the shard names (per server call). Pass -1 to disable this. * @param args The arguments to pass in to the method on the server side. */ protected WorkQueue(INodeProxyManager shardManager, Set allShards, Method method, int shardArrayParamIndex, Object... args) { this(NodeInteraction::new, shardManager, allShards, method, shardArrayParamIndex, args); } /** * Used by unit tests. By providing an alternate factory, this class can be tested without creating * and NodeInteractions. * * @param shardManager The class that maintains the node/shard maps, the node selection * policy, and the node proxies. * @param allShards The entire set of shards for this request. When all these shards * have reported in, the result is complete. * @param method Which method to call on the server side. * @param shardArrayParamIndex Which paramater, if any, should be overwritten with an array of * the shard names (per server call). Pass -1 to disable this. * @param args The arguments to pass in to the method on the server side. */ protected WorkQueue(INodeInteractionFactory interactionFactory, INodeProxyManager shardManager, Set allShards, Method method, int shardArrayParamIndex, Object... args) { if (shardManager == null || allShards == null || method == null) { throw new IllegalArgumentException("Null passed to new WorkQueue()"); } if (allShards.isEmpty()) { throw new IllegalArgumentException("No shards passed to new WorkQueue()"); } this.interactionFactory = interactionFactory; this.shardManager = shardManager; this.method = method; this.shardArrayParamIndex = shardArrayParamIndex; this.args = args != null ? args : new Object[0]; IClosedListener closedListener = new IClosedListener() { public void clientResultClosed() { LOG.trace("Shut down via ClientRequest.close()"); shutdown(); } }; this.results = new ClientResult<>(closedListener, allShards); if (LOG.isTraceEnabled()) { LOG.trace("Creating new " + this); } } /** * Used by unit tests to make toString() output repeatable. */ public static void resetInstanceCounter() { instanceCounter = 0; } /** * Submit a job, which is a call to a server node via an RPC proxy using a NodeInteraction. * Ignored if called after shutdown(), or after result set is closed. * * @param node The node on which to execute the method. * @param nodeShardMap The current node shard map, with failed nodes removed if this is a retry. * @param tryCount This call is the Nth retry. Starts at 1. * @param maxTryCount How often the call should be repeated in maximum. */ public void execute(String node, Map> nodeShardMap, int tryCount, int maxTryCount) { if (!executor.isShutdown() && !results.isClosed()) { if (LOG.isTraceEnabled()) { LOG.trace(String.format("Creating interaction with %s, will use shards: %s, tryCount=%d (id=%d)", node, nodeShardMap.get(node), tryCount, instanceId)); } Runnable interaction = interactionFactory.createInteraction(method, args, shardArrayParamIndex, node, nodeShardMap, tryCount, maxTryCount, shardManager, this, results); if (interaction != null) { try { executor.execute(interaction); } catch (RejectedExecutionException e) { // This could happen, but should be rare. LOG.warn(String.format("Failed to submit node interaction %s (id=%d)", interaction, instanceId)); } } else { LOG.error("Null node interaction runnable for node " + node); } } else { if (LOG.isTraceEnabled()) { LOG.trace(String.format("Not creating interaction with %s, shards=%s, tryCount=%d, executor=%s, result=%s (id=%d)", node, nodeShardMap.get(node), tryCount, executor.isShutdown() ? "shutdown" : "running", results, instanceId)); } } } /** * Stop all threads. Close the result set (making it immutable). * Any calls to execute() after this will be ignored. */ public void shutdown() { if (LOG.isTraceEnabled()) { LOG.trace(String.format("Shutdown() called (id=%d)", instanceId)); } if (!executor.isShutdown()) { executor.shutdownNow(); } if (!results.isClosed()) { results.close(); } } /** * Wait up to timeout msec for the results to be complete (all shards * reporting) then stop the threads and return what we have so far. * * @param timeout maximum msec to wait for. * @return the results of the call, which will be closed. */ public ClientResult getResults(long timeout) { return getResults(new ResultCompletePolicy<>(timeout, true)); } /** * Wait up to timeout msec for the results to be complete (all shards * reporting) then return what we have so far. If shutdown is true, the result * will be closed and any remaining threads will be killed. *

* If you want to do your own polling, pass in 0, true. If you want a simple * all-or-nothing result, pass in N, true, then check isOK() on the result. If * you want to wait for a while then decide for yourself what to do, pass in * N, false (or see IResultPolicy). * * @param timeout maximum msec to wait for. * @param shutdown if true, stops the search. * @return the results of the call, which will be closed. */ public ClientResult getResults(long timeout, boolean shutdown) { return getResults(new ResultCompletePolicy<>(timeout, shutdown)); } /** * Use a user-provided policy to decide how long to wait for and whether to * terminate the call. * * @param policy How to decide when to return and to terminate the call. * @return the results, which may or may not be complete and/or closed. */ public ClientResult getResults(IResultPolicy policy) { int callId = callCounter++; long start = 0; if (LOG.isTraceEnabled()) { LOG.trace(String.format("getResults() policy = %s (id=%d:%d)", policy, instanceId, callId)); start = System.currentTimeMillis(); } long waitTime; while (true) { synchronized (results) { // Need to stay synchronized before waitTime() through wait() or we will // miss notifications. waitTime = policy.waitTime(results); if (waitTime > 0 && !results.isClosed()) { if (LOG.isTraceEnabled()) { LOG.trace(String.format("Waiting %d ms, results = %s (id=%d:%d)", waitTime, results, instanceId, callId)); } try { results.wait(waitTime); } catch (InterruptedException e) { LOG.debug("Interrupted", e); } if (LOG.isTraceEnabled()) { LOG.trace(String.format("Done waiting, results = %s (id=%d:%d)", results, instanceId, callId)); } } else { break; } } } if (waitTime < 0) { if (LOG.isTraceEnabled()) { LOG.trace(String.format("Shutting down work queue, results = %s (id=%d:%d)", results, instanceId, callId)); } executor.shutdownNow(); results.close(); } if (LOG.isTraceEnabled()) { long time = System.currentTimeMillis() - start; LOG.trace(String.format("Returning results = %s, took %d ms (id=%d:%d)", results, time, instanceId, callId)); } return results; } @Override public String toString() { String argsStr = Arrays.asList(args).toString(); argsStr = argsStr.substring(1, argsStr.length() - 1); return String.format("WorkQueue[%s.%s(%s) (id=%d)]", method.getDeclaringClass().getSimpleName(), method.getName(), argsStr, instanceId); } public interface INodeInteractionFactory { Runnable createInteraction(Method method, Object[] args, int shardArrayParamIndex, String node, Map> nodeShardMap, int tryCount, int maxTryCount, INodeProxyManager shardManager, INodeExecutor nodeExecutor, IResultReceiver results); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy