com.kixeye.kixmpp.server.cluster.mapreduce.MapReduceTracker.java Source code

Java tutorial

Introduction

Here is the source code for com.kixeye.kixmpp.server.cluster.mapreduce.MapReduceTracker.java

Source

package com.kixeye.kixmpp.server.cluster.mapreduce;

/*
 * #%L
 * KIXMPP
 * %%
 * Copyright (C) 2014 KIXEYE, Inc
 * %%
 * 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.
 * #L%
 */

import java.util.UUID;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;

import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.RemovalCause;
import com.google.common.cache.RemovalListener;
import com.google.common.cache.RemovalNotification;
import com.kixeye.kixmpp.server.KixmppServer;
import com.kixeye.kixmpp.server.cluster.message.MapReduceRequest;
import com.kixeye.kixmpp.server.cluster.message.MapReduceResponse;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * MapReduceTracker tracks pending MapReduceRequests and completes them when all
 * results are available or they timeout.
 */
public class MapReduceTracker implements RemovalListener<UUID, MapReduceTracker.RequestWrapper> {
    private final Logger log = LoggerFactory.getLogger(MapReduceTracker.class);

    private final KixmppServer server;
    private final Cache<UUID, RequestWrapper> requests;

    public MapReduceTracker(KixmppServer server, ScheduledExecutorService scheduledExecutorService) {
        this.server = server;
        this.requests = CacheBuilder.newBuilder().expireAfterWrite(15, TimeUnit.SECONDS).removalListener(this)
                .build();

        // periodically call cache clean up expiration callbacks
        scheduledExecutorService.scheduleWithFixedDelay(new Runnable() {
            @Override
            public void run() {
                requests.cleanUp();
            }
        }, 10, 10, TimeUnit.SECONDS);
    }

    @Override
    public void onRemoval(RemovalNotification<UUID, MapReduceTracker.RequestWrapper> notification) {
        if (notification.getCause() == RemovalCause.EXPIRED) {
            RequestWrapper wrapper = notification.getValue();
            log.warn("Timing out MapReduce request <{}> with ref count <{}>", wrapper.getClass().toString(),
                    wrapper.pendingResponseCount.get());
            wrapper.request.onComplete(true);
        }
    }

    /**
     * Send MapReduceRequest to all nodes in the cluster, including
     * the local node.
     *
     * @param request
     */
    public void sendRequest(MapReduceRequest request) {
        UUID transId = UUID.randomUUID();
        request.setTransactionId(transId);
        requests.put(transId, new RequestWrapper(request, server.getCluster().getNodeCount()));
        server.getCluster().sendMessageToAll(request, true);
    }

    /**
     * Process 1 of N responses to a request, possibly completing it.
     *
     * @param response
     */
    public void processResponse(MapReduceResponse response) {
        UUID transId = response.getTransactionId();
        RequestWrapper wrapper = requests.getIfPresent(transId);
        if (wrapper != null) {
            wrapper.request.mergeResponse(response);
            if (wrapper.pendingResponseCount.decrementAndGet() == 0) {
                requests.invalidate(transId);
                if (log.isDebugEnabled()) {
                    log.debug("Completing MapReduce request <{}> in {} ms", wrapper.request.getClass().toString(),
                            (System.currentTimeMillis() - wrapper.getStartTime()));
                }
                wrapper.request.onComplete(false);
            }
        }
    }

    /**
     * Wrapper class to hold the request and pending response count.
     */
    public static class RequestWrapper {
        private MapReduceRequest request;
        private AtomicInteger pendingResponseCount;
        private long startTime;

        public RequestWrapper(MapReduceRequest request, int pendingResponseCount) {
            this.request = request;
            this.pendingResponseCount = new AtomicInteger(pendingResponseCount);
            this.startTime = System.currentTimeMillis();
        }

        public long getStartTime() {
            return startTime;
        }
    }
}