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

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

/**
 * 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.util.ChokException;
import org.apache.hadoop.ipc.VersionedProtocol;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
import java.util.List;
import java.util.Map;

/**
 * This class is responsible for calling the sever node via an RPC proxy, and
 * possibly scheduling retires if errors occur. Only M-1 retries are attempted,
 * whereby M is the given maxRetryCount (M calls total). With replication level
 * N, there can be at most N-1 retries (N calls total).
 */
class NodeInteraction implements Runnable {

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

    // Used to make logs easier to read.
    private static int interactionInstanceCounter;

    private final Method method;
    private final Object[] args;
    private final int shardArrayIndex;
    private final String node;
    private final Map> node2ShardsMap;
    private final List shards;
    private final int tryCount;
    private final int maxTryCount;
    private final INodeExecutor workQueue;
    private final INodeProxyManager shardManager;
    private final IResultReceiver result;
    private final int instanceId = interactionInstanceCounter++;

    /**
     * Create a node interaction. This will make one call to one node, listing
     * multiple shards.
     *
     * @param method          Which method to call on the server. This method must have come
     *                        from the same interface used to creat the RPC proxy in the first
     *                        place (see Client constructor).
     * @param args            The arguments to pass to the method. When calling the server, the
     *                        shard list argument will be modified if shardArrayIndex >= 0.
     * @param shardArrayIndex Which parameter, if any, to overwrite with a String[] of shard
     *                        names (this one arg then changes an a per-node basis, otherwise
     *                        all arguments are the same for all calls to nodes). This is
     *                        optional, depending on the needs of the server. When < 0, no
     *                        overwriting is done.
     * @param node            The name of the node to contact. This is used to get the node's
     *                        proxy object (see IShardProxyManager).
     * @param node2ShardsMap  The mapping from nodes to shards for all nodes. Used initially
     *                        with the node "node", but other nodes if errors occur and we
     *                        retry. For every retry the failed node is removed from the map and
     *                        the map is then used to submit a retry job.
     * @param tryCount        This interaction is the Nth retry (starts at 1). We use this to
     *                        decide if we should retry shards.
     * @param maxTryCount     The maximum number a interaction is tried to be executed.
     * @param shardManager    Our source of node proxies, and access to a node selection policy
     *                        to pick the nodes to use for retires. Also we notify this object
     *                        on node failures.
     * @param workQueue       Use this if we need to resubmit a retry job. Will result in a new
     *                        NodeInteraction.
     * @param result The destination to write to. If we get a result from the node we
     * add it. If we get an error and submit retries we do not use it (the
     * retry jobs will write to it for us). If we get an error and do not
     * retry we write the error to it.
     */
    public NodeInteraction(Method method, Object[] args, int shardArrayIndex, String node, Map> node2ShardsMap, int tryCount, int maxTryCount, INodeProxyManager shardManager, INodeExecutor workQueue, IResultReceiver result) {
        this.method = method;
        // Make a copy in case we will be modifying the shard list.
        this.args = Arrays.copyOf(args, args.length);
        this.shardArrayIndex = shardArrayIndex;
        this.node = node;
        this.node2ShardsMap = node2ShardsMap;
        shards = node2ShardsMap.get(node);
        this.tryCount = tryCount;
        this.maxTryCount = maxTryCount;
        this.workQueue = workQueue;
        this.shardManager = shardManager;
        this.result = result;
    }

    @SuppressWarnings("unchecked")
    public void run() {
        String methodDesc = null;
        try {
            VersionedProtocol proxy = shardManager.getProxy(node, false);
            if (proxy == null) {
                throw new ChokException("No proxy for node: " + node);
            }
            if (shardArrayIndex >= 0) {
                // We need to pass the list of shards to the server's method.
                args[shardArrayIndex] = shards.toArray(new String[shards.size()]);
            }
            long startTime = 0;
            if (LOG.isTraceEnabled()) {
                methodDesc = describeMethodCall(method, args, node);
                LOG.trace(String.format("About to invoke %s using proxy %s (id=%d)", methodDesc, Proxy.getInvocationHandler(proxy), instanceId));
                startTime = System.currentTimeMillis();
            }
            T result = (T) method.invoke(proxy, args);
            shardManager.reportNodeCommunicationSuccess(node);
            if (LOG.isTraceEnabled()) {
                LOG.trace(String.format("Calling %s returned %s, took %d msec (id=%d)", methodDesc, resultToString(result), (System.currentTimeMillis() - startTime), instanceId));
                String methodDesc2 = describeMethodCall(method, args, node);
                if (!methodDesc.equals(methodDesc2)) {
                    LOG.error(String.format("Method call changed from %s to %s (id=%d)", methodDesc, methodDesc2, instanceId));
                }
            }
            this.result.addResult(result, shards);
        } catch (Throwable t) {
            // Notify the work queue, so it can mark the node as down.
            shardManager.reportNodeCommunicationFailure(node, t);
            if (tryCount >= maxTryCount) {
                LOG.error(String.format("Error calling %s (try # %d of %d) (id=%d)", (methodDesc != null ? methodDesc : method + " on " + node), tryCount, maxTryCount, instanceId), t);
                result.addError(new ChokException(String.format("%s for shards %s failed (id=%d)", getClass().getSimpleName(), shards, instanceId), t), shards);
                return;
            }
            if (!result.isClosed()) {
                try {
                    // Find new node(s) for our shards and add to global node2ShardMap
                    Map> retryMap = shardManager.createNode2ShardsMap(node2ShardsMap.get(node));
                    LOG.warn(String.format("Failed to interact with node %s. Trying with other node(s) %s (id=%d)", node, retryMap.keySet(), instanceId), t);
                    // Execute the action again for every node
                    for (String newNode : retryMap.keySet()) {
                        workQueue.execute(newNode, retryMap, tryCount + 1, maxTryCount);
                    }
                } catch (ShardAccessException e) {
                    LOG.error(String.format("Error calling %s (try # %d of %d) (id=%d)", (methodDesc != null ? methodDesc : method + " on " + node), tryCount, maxTryCount, instanceId), t);
                    result.addError(e, shards);
                }
            }
            // We have no results to report. Submitted jobs will hopefully get results
            // instead.
        }
    }

    private String describeMethodCall(Method method, Object[] args, String nodeName) {
        StringBuilder builder = new StringBuilder(method.getDeclaringClass().getSimpleName());
        builder.append(".");
        builder.append(method.getName());
        builder.append("(");
        String sep = "";
        for (Object arg : args) {
            builder.append(sep);
            if (arg == null) {
                builder.append("null");
            } else if (arg instanceof String[]) {
                // TODO: all array types, lists, maps.
                String[] strs = (String[]) arg;
                String sep2 = "";
                builder.append("[");
                for (String str : strs) {
                    builder.append(sep2).append("\"").append(str).append("\"");
                    sep2 = ", ";
                }
                builder.append("]");
            } else {
                builder.append(arg.toString());
            }
            sep = ", ";
        }
        builder.append(") on ");
        builder.append(nodeName);
        return builder.toString();
    }

    private String resultToString(T result) {
        String s = "null";
        if (result != null) {
            try {
                s = result.toString();
            } catch (Throwable t) {
                LOG.trace("Error calling toString() on result", t);
                s = "(toString() err)";
            }
        }
        if (s == null) {
            s = "(null toString())";
        }
        return s;
    }

    @Override
    public String toString() {
        return "NodeInteraction: call " + method.getName() + " on " + node;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy