/* * Copyright (c) 2014 Cisco 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.mdsal.binding.dom.adapter; import com.google.common.base.Preconditions; import com.google.common.util.concurrent.FluentFuture; import com.google.common.util.concurrent.FutureCallback; import com.google.common.util.concurrent.Futures; import com.google.common.util.concurrent.ListenableFuture; import com.google.common.util.concurrent.MoreExecutors; import org.eclipse.jdt.annotation.NonNull; import org.opendaylight.mdsal.binding.api.BindingTransactionChain; import org.opendaylight.mdsal.binding.api.ReadTransaction; import org.opendaylight.mdsal.binding.api.ReadWriteTransaction; import org.opendaylight.mdsal.binding.api.WriteTransaction; import org.opendaylight.mdsal.common.api.AsyncTransaction; import org.opendaylight.mdsal.common.api.CommitInfo; import org.opendaylight.mdsal.common.api.TransactionChain; import org.opendaylight.mdsal.common.api.TransactionChainListener; import org.opendaylight.mdsal.dom.api.DOMDataBroker; import org.opendaylight.mdsal.dom.api.DOMDataTreeReadTransaction; import org.opendaylight.mdsal.dom.api.DOMDataTreeReadWriteTransaction; import org.opendaylight.mdsal.dom.api.DOMDataTreeWriteTransaction; import org.opendaylight.mdsal.dom.api.DOMTransactionChain; import org.opendaylight.yangtools.concepts.Delegator; import org.slf4j.Logger; import org.slf4j.LoggerFactory; final class BindingDOMTransactionChainAdapter implements BindingTransactionChain, Delegator { private static final Logger LOG = LoggerFactory.getLogger(BindingDOMTransactionChainAdapter.class); private final DOMTransactionChain delegate; private final BindingToNormalizedNodeCodec codec; private final DelegateChainListener domListener; private final TransactionChainListener bindingListener; BindingDOMTransactionChainAdapter(final DOMDataBroker chainFactory, final BindingToNormalizedNodeCodec codec, final TransactionChainListener listener) { Preconditions.checkNotNull(chainFactory, "DOM Transaction chain factory must not be null"); this.domListener = new DelegateChainListener(); this.bindingListener = listener; this.delegate = chainFactory.createTransactionChain(domListener); this.codec = codec; } @Override public DOMTransactionChain getDelegate() { return delegate; } @Override public ReadTransaction newReadOnlyTransaction() { final DOMDataTreeReadTransaction delegateTx = delegate.newReadOnlyTransaction(); return new BindingDOMReadTransactionAdapter(delegateTx, codec); } @Override public WriteTransaction newWriteOnlyTransaction() { final DOMDataTreeWriteTransaction delegateTx = delegate.newWriteOnlyTransaction(); return new BindingDOMWriteTransactionAdapter(delegateTx, codec) { @Override public @NonNull FluentFuture commit() { return listenForFailure(this, super.commit()); } }; } @Override public ReadWriteTransaction newReadWriteTransaction() { final DOMDataTreeReadWriteTransaction delegateTx = delegate.newReadWriteTransaction(); return new BindingDOMReadWriteTransactionAdapter(delegateTx, codec) { @Override public @NonNull FluentFuture commit() { return listenForFailure(this, super.commit()); } }; } private > F listenForFailure(final WriteTransaction tx, final F future) { Futures.addCallback(future, new FutureCallback() { @Override public void onFailure(final Throwable throwable) { failTransactionChain(tx, throwable); } @Override public void onSuccess(final T result) { // Intentionally NOOP } }, MoreExecutors.directExecutor()); return future; } private void failTransactionChain(final WriteTransaction tx, final Throwable throwable) { /* * We asume correct state change for underlaying transaction * * chain, so we are not changing any of our internal state * to mark that we failed. */ this.bindingListener.onTransactionChainFailed(this, tx, throwable); } @Override public void close() { delegate.close(); } private final class DelegateChainListener implements TransactionChainListener { @Override public void onTransactionChainFailed(final TransactionChain chain, final AsyncTransaction transaction, final Throwable cause) { Preconditions.checkState(delegate.equals(chain), "Illegal state - listener for %s was invoked for incorrect chain %s.", delegate, chain); /* * Intentionally NOOP, callback for failure, since we * are also listening on each transaction future for failure, * in order to have reference to Binding Transaction (which was seen by client * of this transaction chain), instead of DOM transaction * which is known only to this chain, binding transaction implementation * and underlying transaction chain. * */ LOG.debug("Transaction chain {} failed. Failed DOM Transaction {}",this,transaction,cause); } @Override public void onTransactionChainSuccessful(final TransactionChain chain) { Preconditions.checkState(delegate.equals(chain), "Illegal state - listener for %s was invoked for incorrect chain %s.", delegate, chain); bindingListener.onTransactionChainSuccessful(BindingDOMTransactionChainAdapter.this); } } }