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

org.redisson.core.RedissonMultiLock Maven / Gradle / Ivy

There is a newer version: 3.40.2
Show newest version
/**
 * Copyright 2014 Nikita Koksharov, Nickolay Borbit
 *
 * 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 org.redisson.core;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;

import io.netty.util.concurrent.DefaultPromise;
import io.netty.util.concurrent.Future;
import io.netty.util.concurrent.FutureListener;
import io.netty.util.concurrent.Promise;
import io.netty.util.internal.PlatformDependent;

/**
 * Guarantees multiple locks operation handling (lock, tryLock...)
 * in atomic way without deadlocks.
 *
 * @author Nikita Koksharov
 *
 */
public class RedissonMultiLock implements Lock {

    final List locks = new ArrayList();

    /**
     * Creates instance with multiple {@link RLock} objects.
     * Each RLock object could be created by own Redisson instance.
     *
     * @param locks
     */
    public RedissonMultiLock(RLock... locks) {
        if (locks.length == 0) {
            throw new IllegalArgumentException("Lock objects are not defined");
        }
        this.locks.addAll(Arrays.asList(locks));
    }

    @Override
    public void lock() {
        try {
            lockInterruptibly();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }

    public void lock(long leaseTime, TimeUnit unit) {
        try {
            lockInterruptibly(leaseTime, unit);
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }

    public void lockInterruptibly(long leaseTime, TimeUnit unit) throws InterruptedException {
        final CountDownLatch latch = new CountDownLatch(1);
        final AtomicReference result = new AtomicReference();
        Promise promise = new DefaultPromise() {
            public Promise setSuccess(Void result) {
                latch.countDown();
                return this;
            };

            public Promise setFailure(Throwable cause) {
                result.set(cause);
                latch.countDown();
                return this;
            };
        };

        lock(promise, 0, leaseTime, unit);

        latch.await();
        if (result.get() instanceof Throwable) {
            PlatformDependent.throwException((Throwable)result.get());
        }
    }

    @Override
    public void lockInterruptibly() throws InterruptedException {
        final CountDownLatch latch = new CountDownLatch(1);
        final AtomicReference result = new AtomicReference();
        Promise promise = new DefaultPromise() {
            public Promise setSuccess(Void result) {
                latch.countDown();
                return this;
            };

            public Promise setFailure(Throwable cause) {
                result.set(cause);
                latch.countDown();
                return this;
            };
        };

        lock(promise, 0, -1, null);

        latch.await();
        if (result.get() instanceof Throwable) {
            PlatformDependent.throwException((Throwable)result.get());
        }
    }

    private void lock(final Promise promise, final long waitTime, final long leaseTime, final TimeUnit unit) throws InterruptedException {
        final AtomicInteger tryLockRequestsAmount = new AtomicInteger();
        final Map, RLock> tryLockFutures = new HashMap, RLock>(locks.size());

        FutureListener listener = new FutureListener() {

            AtomicBoolean unlock = new AtomicBoolean();

            @Override
            public void operationComplete(Future future) throws Exception {
                if (!future.isSuccess()) {
                    // unlock once
                    if (unlock.compareAndSet(false, true)) {
                        for (RLock lock : locks) {
                            lock.unlockAsync();
                        }

                        promise.setFailure(future.cause());
                    }
                    return;
                }

                Boolean res = future.getNow();
                // unlock once
                if (!res && unlock.compareAndSet(false, true)) {
                    for (RLock lock : locks) {
                        lock.unlockAsync();
                    }

                    RLock lock = tryLockFutures.get(future);
                    lock.lockAsync().addListener(new FutureListener() {
                        @Override
                        public void operationComplete(Future future) throws Exception {
                            if (!future.isSuccess()) {
                                promise.setFailure(future.cause());
                                return;
                            }

                            lock(promise, waitTime, leaseTime, unit);
                        }
                    });
                }
                if (!unlock.get() && tryLockRequestsAmount.decrementAndGet() == 0) {
                    promise.setSuccess(null);
                }
            }
        };

        for (RLock lock : locks) {
            if (lock.isHeldByCurrentThread()) {
                continue;
            }

            tryLockRequestsAmount.incrementAndGet();
            if (waitTime > 0 || leaseTime > 0) {
                tryLockFutures.put(lock.tryLockAsync(waitTime, leaseTime, unit), lock);
            } else {
                tryLockFutures.put(lock.tryLockAsync(), lock);
            }
        }

        for (Future future : tryLockFutures.keySet()) {
            future.addListener(listener);
        }
    }

    @Override
    public boolean tryLock() {
        List> tryLockFutures = new ArrayList>(locks.size());
        for (RLock lock : locks) {
            tryLockFutures.add(lock.tryLockAsync());
        }

        for (Future future : tryLockFutures) {
            try {
                if (!future.syncUninterruptibly().getNow()) {
                    unlockInner();
                    return false;
                }
            } catch (RuntimeException e) {
                unlockInner();
                throw e;
            }
        }

        return true;
    }

    private void unlockInner() {
        List> futures = new ArrayList>(locks.size());
        for (RLock lock : locks) {
            futures.add(lock.unlockAsync());
        }

        for (Future unlockFuture : futures) {
            unlockFuture.awaitUninterruptibly();
        }
    }

    @Override
    public boolean tryLock(long waitTime, TimeUnit unit) throws InterruptedException {
        return tryLock(waitTime, -1, unit);
    }

    public boolean tryLock(long waitTime, long leaseTime, TimeUnit unit) throws InterruptedException {
        List> tryLockFutures = new ArrayList>(locks.size());
        for (RLock lock : locks) {
            tryLockFutures.add(lock.tryLockAsync(waitTime, leaseTime, unit));
        }

        for (Future future : tryLockFutures) {
            try {
                if (!future.syncUninterruptibly().getNow()) {
                    unlockInner();
                    return false;
                }
            } catch (RuntimeException e) {
                unlockInner();
                throw e;
            }
        }

        return true;
    }


    @Override
    public void unlock() {
        List> futures = new ArrayList>(locks.size());

        for (RLock lock : locks) {
            futures.add(lock.unlockAsync());
        }

        for (Future future : futures) {
            future.syncUninterruptibly();
        }
    }


    @Override
    public Condition newCondition() {
        throw new UnsupportedOperationException();
    }

}