Refactor DOMDataBrokerImpl
[controller.git] / opendaylight / md-sal / sal-dom-broker / src / main / java / org / opendaylight / controller / md / sal / dom / broker / impl / DOMConcurrentDataCommitCoordinator.java
diff --git a/opendaylight/md-sal/sal-dom-broker/src/main/java/org/opendaylight/controller/md/sal/dom/broker/impl/DOMConcurrentDataCommitCoordinator.java b/opendaylight/md-sal/sal-dom-broker/src/main/java/org/opendaylight/controller/md/sal/dom/broker/impl/DOMConcurrentDataCommitCoordinator.java
deleted file mode 100644 (file)
index d8a7a00..0000000
+++ /dev/null
@@ -1,348 +0,0 @@
-/*
- * Copyright (c) 2014 Brocade Communications Systems, Inc. and others.  All rights reserved.
- *
- * This program and the accompanying materials are made available under the
- * terms of the Eclipse Public License v1.0 which accompanies this distribution,
- * and is available at http://www.eclipse.org/legal/epl-v10.html
- */
-package org.opendaylight.controller.md.sal.dom.broker.impl;
-
-import com.google.common.base.Preconditions;
-import com.google.common.collect.Iterables;
-import com.google.common.util.concurrent.AbstractFuture;
-import com.google.common.util.concurrent.AbstractListeningExecutorService;
-import com.google.common.util.concurrent.CheckedFuture;
-import com.google.common.util.concurrent.FutureCallback;
-import com.google.common.util.concurrent.Futures;
-import com.google.common.util.concurrent.ListenableFuture;
-import java.util.List;
-import java.util.concurrent.Executor;
-import java.util.concurrent.ExecutorService;
-import java.util.concurrent.TimeUnit;
-import java.util.concurrent.atomic.AtomicInteger;
-import org.opendaylight.controller.md.sal.common.api.data.TransactionCommitFailedException;
-import org.opendaylight.controller.md.sal.dom.api.DOMDataWriteTransaction;
-import org.opendaylight.controller.sal.core.spi.data.DOMStoreThreePhaseCommitCohort;
-import org.opendaylight.yangtools.util.DurationStatisticsTracker;
-import org.opendaylight.yangtools.util.concurrent.MappingCheckedFuture;
-import org.slf4j.Logger;
-import org.slf4j.LoggerFactory;
-
-/**
- * Implementation of DOMDataCommitExecutor that coordinates transaction commits concurrently. The 3
- * commit phases (canCommit, preCommit, and commit) are performed serially and non-blocking
- * (ie async) per transaction but multiple transaction commits can run concurrent.
- *
- * @author Thomas Pantelis
- */
-public class DOMConcurrentDataCommitCoordinator implements DOMDataCommitExecutor {
-
-    private static final String CAN_COMMIT = "CAN_COMMIT";
-    private static final String PRE_COMMIT = "PRE_COMMIT";
-    private static final String COMMIT = "COMMIT";
-
-    private static final Logger LOG = LoggerFactory.getLogger(DOMConcurrentDataCommitCoordinator.class);
-
-    private final DurationStatisticsTracker commitStatsTracker = DurationStatisticsTracker.createConcurrent();
-
-    /**
-     * This executor is used to execute Future listener callback Runnables async.
-     */
-    private final ExecutorService clientFutureCallbackExecutor;
-
-    /**
-     * This executor is re-used internally in calls to Futures#addCallback to avoid the overhead
-     * of Futures#addCallback creating a MoreExecutors#sameThreadExecutor for each call.
-     */
-    private final ExecutorService internalFutureCallbackExecutor = new SimpleSameThreadExecutor();
-
-    public DOMConcurrentDataCommitCoordinator(ExecutorService listenableFutureExecutor) {
-        this.clientFutureCallbackExecutor = Preconditions.checkNotNull(listenableFutureExecutor);
-    }
-
-    public DurationStatisticsTracker getCommitStatsTracker() {
-        return commitStatsTracker;
-    }
-
-    @Override
-    public CheckedFuture<Void, TransactionCommitFailedException> submit(DOMDataWriteTransaction transaction,
-            Iterable<DOMStoreThreePhaseCommitCohort> cohorts) {
-
-        Preconditions.checkArgument(transaction != null, "Transaction must not be null.");
-        Preconditions.checkArgument(cohorts != null, "Cohorts must not be null.");
-        LOG.debug("Tx: {} is submitted for execution.", transaction.getIdentifier());
-
-        final int cohortSize = Iterables.size(cohorts);
-        final AsyncNotifyingSettableFuture clientSubmitFuture =
-                new AsyncNotifyingSettableFuture(clientFutureCallbackExecutor);
-
-        doCanCommit(clientSubmitFuture, transaction, cohorts, cohortSize);
-
-        return MappingCheckedFuture.create(clientSubmitFuture,
-                TransactionCommitFailedExceptionMapper.COMMIT_ERROR_MAPPER);
-    }
-
-    private void doCanCommit(final AsyncNotifyingSettableFuture clientSubmitFuture,
-            final DOMDataWriteTransaction transaction,
-            final Iterable<DOMStoreThreePhaseCommitCohort> cohorts, final int cohortSize) {
-
-        final long startTime = System.nanoTime();
-
-        // Not using Futures.allAsList here to avoid its internal overhead.
-        final AtomicInteger remaining = new AtomicInteger(cohortSize);
-        FutureCallback<Boolean> futureCallback = new FutureCallback<Boolean>() {
-            @Override
-            public void onSuccess(Boolean result) {
-                if (result == null || !result) {
-                    handleException(clientSubmitFuture, transaction, cohorts, cohortSize,
-                            CAN_COMMIT, TransactionCommitFailedExceptionMapper.CAN_COMMIT_ERROR_MAPPER,
-                            new TransactionCommitFailedException(
-                                            "Can Commit failed, no detailed cause available."));
-                } else {
-                    if(remaining.decrementAndGet() == 0) {
-                        // All cohorts completed successfully - we can move on to the preCommit phase
-                        doPreCommit(startTime, clientSubmitFuture, transaction, cohorts, cohortSize);
-                    }
-                }
-            }
-
-            @Override
-            public void onFailure(Throwable t) {
-                handleException(clientSubmitFuture, transaction, cohorts, cohortSize, CAN_COMMIT,
-                        TransactionCommitFailedExceptionMapper.CAN_COMMIT_ERROR_MAPPER, t);
-            }
-        };
-
-        for(DOMStoreThreePhaseCommitCohort cohort: cohorts) {
-            ListenableFuture<Boolean> canCommitFuture = cohort.canCommit();
-            Futures.addCallback(canCommitFuture, futureCallback, internalFutureCallbackExecutor);
-        }
-    }
-
-    private void doPreCommit(final long startTime, final AsyncNotifyingSettableFuture clientSubmitFuture,
-            final DOMDataWriteTransaction transaction,
-            final Iterable<DOMStoreThreePhaseCommitCohort> cohorts, final int cohortSize) {
-
-        // Not using Futures.allAsList here to avoid its internal overhead.
-        final AtomicInteger remaining = new AtomicInteger(cohortSize);
-        FutureCallback<Void> futureCallback = new FutureCallback<Void>() {
-            @Override
-            public void onSuccess(Void notUsed) {
-                if(remaining.decrementAndGet() == 0) {
-                    // All cohorts completed successfully - we can move on to the commit phase
-                    doCommit(startTime, clientSubmitFuture, transaction, cohorts, cohortSize);
-                }
-            }
-
-            @Override
-            public void onFailure(Throwable t) {
-                handleException(clientSubmitFuture, transaction, cohorts, cohortSize, PRE_COMMIT,
-                        TransactionCommitFailedExceptionMapper.PRE_COMMIT_MAPPER, t);
-            }
-        };
-
-        for(DOMStoreThreePhaseCommitCohort cohort: cohorts) {
-            ListenableFuture<Void> preCommitFuture = cohort.preCommit();
-            Futures.addCallback(preCommitFuture, futureCallback, internalFutureCallbackExecutor);
-        }
-    }
-
-    private void doCommit(final long startTime, final AsyncNotifyingSettableFuture clientSubmitFuture,
-            final DOMDataWriteTransaction transaction,
-            final Iterable<DOMStoreThreePhaseCommitCohort> cohorts, final int cohortSize) {
-
-        // Not using Futures.allAsList here to avoid its internal overhead.
-        final AtomicInteger remaining = new AtomicInteger(cohortSize);
-        FutureCallback<Void> futureCallback = new FutureCallback<Void>() {
-            @Override
-            public void onSuccess(Void notUsed) {
-                if(remaining.decrementAndGet() == 0) {
-                    // All cohorts completed successfully - we're done.
-                    commitStatsTracker.addDuration(System.nanoTime() - startTime);
-
-                    clientSubmitFuture.set();
-                }
-            }
-
-            @Override
-            public void onFailure(Throwable t) {
-                handleException(clientSubmitFuture, transaction, cohorts, cohortSize, COMMIT,
-                        TransactionCommitFailedExceptionMapper.COMMIT_ERROR_MAPPER, t);
-            }
-        };
-
-        for(DOMStoreThreePhaseCommitCohort cohort: cohorts) {
-            ListenableFuture<Void> commitFuture = cohort.commit();
-            Futures.addCallback(commitFuture, futureCallback, internalFutureCallbackExecutor);
-        }
-    }
-
-    private void handleException(final AsyncNotifyingSettableFuture clientSubmitFuture,
-            final DOMDataWriteTransaction transaction,
-            final Iterable<DOMStoreThreePhaseCommitCohort> cohorts, int cohortSize,
-            final String phase, final TransactionCommitFailedExceptionMapper exMapper,
-            final Throwable t) {
-
-        if(clientSubmitFuture.isDone()) {
-            // We must have had failures from multiple cohorts.
-            return;
-        }
-
-        LOG.warn("Tx: {} Error during phase {}, starting Abort", transaction.getIdentifier(), phase, t);
-        Exception e;
-        if(t instanceof Exception) {
-            e = (Exception)t;
-        } else {
-            e = new RuntimeException("Unexpected error occurred", t);
-        }
-
-        final TransactionCommitFailedException clientException = exMapper.apply(e);
-
-        // Transaction failed - tell all cohorts to abort.
-
-        @SuppressWarnings("unchecked")
-        ListenableFuture<Void>[] canCommitFutures = new ListenableFuture[cohortSize];
-        int i = 0;
-        for(DOMStoreThreePhaseCommitCohort cohort: cohorts) {
-            canCommitFutures[i++] = cohort.abort();
-        }
-
-        ListenableFuture<List<Void>> combinedFuture = Futures.allAsList(canCommitFutures);
-        Futures.addCallback(combinedFuture, new FutureCallback<List<Void>>() {
-            @Override
-            public void onSuccess(List<Void> notUsed) {
-                // Propagate the original exception to the client.
-                clientSubmitFuture.setException(clientException);
-            }
-
-            @Override
-            public void onFailure(Throwable t) {
-                LOG.error("Tx: {} Error during Abort.", transaction.getIdentifier(), t);
-
-                // Propagate the original exception as that is what caused the Tx to fail and is
-                // what's interesting to the client.
-                clientSubmitFuture.setException(clientException);
-            }
-        }, internalFutureCallbackExecutor);
-    }
-
-    /**
-     * A settable future that uses an {@link Executor} to execute listener callback Runnables,
-     * registered via {@link #addListener}, asynchronously when this future completes. This is
-     * done to guarantee listener executions are off-loaded onto another thread to avoid blocking
-     * the thread that completed this future, as a common use case is to pass an executor that runs
-     * tasks in the same thread as the caller (ie MoreExecutors#sameThreadExecutor)
-     * to {@link #addListener}.
-     *
-     * FIXME: This class should probably be moved to yangtools common utils for re-usability and
-     * unified with AsyncNotifyingListenableFutureTask.
-     */
-    private static class AsyncNotifyingSettableFuture extends AbstractFuture<Void> {
-
-        /**
-         * ThreadLocal used to detect if the task completion thread is running the future listener Runnables.
-         */
-        private static final ThreadLocal<Boolean> ON_TASK_COMPLETION_THREAD_TL = new ThreadLocal<Boolean>();
-
-        private final ExecutorService listenerExecutor;
-
-        AsyncNotifyingSettableFuture(ExecutorService listenerExecutor) {
-            this.listenerExecutor = listenerExecutor;
-        }
-
-        @Override
-        public void addListener(final Runnable listener, final Executor executor) {
-            // Wrap the listener Runnable in a DelegatingRunnable. If the specified executor is one
-            // that runs tasks in the same thread as the caller submitting the task
-            // (e.g. {@link com.google.common.util.concurrent.MoreExecutors#sameThreadExecutor}) and
-            // the listener is executed from the #set methods, then the DelegatingRunnable will detect
-            // this via the ThreadLocal and submit the listener Runnable to the listenerExecutor.
-            //
-            // On the other hand, if this task is already complete, the call to ExecutionList#add in
-            // superclass will execute the listener Runnable immediately and, since the ThreadLocal
-            // won't be set, the DelegatingRunnable will run the listener Runnable inline.
-            super.addListener(new DelegatingRunnable(listener, listenerExecutor), executor);
-        }
-
-        boolean set() {
-            ON_TASK_COMPLETION_THREAD_TL.set(Boolean.TRUE);
-            try {
-                return super.set(null);
-            } finally {
-                ON_TASK_COMPLETION_THREAD_TL.set(null);
-            }
-        }
-
-        @Override
-        protected boolean setException(Throwable throwable) {
-            ON_TASK_COMPLETION_THREAD_TL.set(Boolean.TRUE);
-            try {
-                return super.setException(throwable);
-            } finally {
-                ON_TASK_COMPLETION_THREAD_TL.set(null);
-            }
-        }
-
-        private static final class DelegatingRunnable implements Runnable {
-            private final Runnable delegate;
-            private final Executor executor;
-
-            DelegatingRunnable(final Runnable delegate, final Executor executor) {
-                this.delegate = Preconditions.checkNotNull(delegate);
-                this.executor = Preconditions.checkNotNull(executor);
-            }
-
-            @Override
-            public void run() {
-                if (ON_TASK_COMPLETION_THREAD_TL.get() != null) {
-                    // We're running on the task completion thread so off-load to the executor.
-                    LOG.trace("Submitting ListenenableFuture Runnable from thread {} to executor {}",
-                            Thread.currentThread().getName(), executor);
-                    executor.execute(delegate);
-                } else {
-                    // We're not running on the task completion thread so run the delegate inline.
-                    LOG.trace("Executing ListenenableFuture Runnable on this thread: {}",
-                            Thread.currentThread().getName());
-                    delegate.run();
-                }
-            }
-        }
-    }
-
-    /**
-     * A simple same-thread executor without the internal locking overhead that
-     * MoreExecutors#sameThreadExecutor has. The #execute method is the only one of concern - we
-     * don't shutdown the executor so the other methods irrelevant.
-     */
-    private static class SimpleSameThreadExecutor extends AbstractListeningExecutorService {
-
-        @Override
-        public void execute(Runnable command) {
-            command.run();
-        }
-
-        @Override
-        public boolean awaitTermination(long arg0, TimeUnit arg1) throws InterruptedException {
-            return true;
-        }
-
-        @Override
-        public boolean isShutdown() {
-            return false;
-        }
-
-        @Override
-        public boolean isTerminated() {
-            return false;
-        }
-
-        @Override
-        public void shutdown() {
-        }
-
-        @Override
-        public List<Runnable> shutdownNow() {
-            return null;
-        }
-    }
-}