001/*
002 * Licensed to the Apache Software Foundation (ASF) under one or more
003 * contributor license agreements.  See the NOTICE file distributed with
004 * this work for additional information regarding copyright ownership.
005 * The ASF licenses this file to You under the Apache License, Version 2.0
006 * (the "License"); you may not use this file except in compliance with
007 * the License.  You may obtain a copy of the License at
008 *
009 *      http://www.apache.org/licenses/LICENSE-2.0
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS,
013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014 * See the License for the specific language governing permissions and
015 * limitations under the License.
016 */
017package org.apache.commons.lang3.concurrent;
018
019import java.util.concurrent.ConcurrentMap;
020import java.util.concurrent.ExecutionException;
021import java.util.concurrent.Future;
022import java.util.concurrent.TimeUnit;
023
024import org.apache.commons.lang3.Validate;
025import org.apache.commons.lang3.exception.ExceptionUtils;
026
027/**
028 * An utility class providing functionality related to the {@code
029 * java.util.concurrent} package.
030 *
031 * @since 3.0
032 */
033public class ConcurrentUtils {
034
035    /**
036     * Private constructor so that no instances can be created. This class
037     * contains only static utility methods.
038     */
039    private ConcurrentUtils() {
040    }
041
042    /**
043     * Inspects the cause of the specified {@link ExecutionException} and
044     * creates a {@link ConcurrentException} with the checked cause if
045     * necessary. This method performs the following checks on the cause of the
046     * passed in exception:
047     * <ul>
048     * <li>If the passed in exception is <b>null</b> or the cause is
049     * <b>null</b>, this method returns <b>null</b>.</li>
050     * <li>If the cause is a runtime exception, it is directly thrown.</li>
051     * <li>If the cause is an error, it is directly thrown, too.</li>
052     * <li>In any other case the cause is a checked exception. The method then
053     * creates a {@link ConcurrentException}, initializes it with the cause, and
054     * returns it.</li>
055     * </ul>
056     *
057     * @param ex the exception to be processed
058     * @return a {@link ConcurrentException} with the checked cause
059     */
060    public static ConcurrentException extractCause(final ExecutionException ex) {
061        if (ex == null || ex.getCause() == null) {
062            return null;
063        }
064        ExceptionUtils.throwUnchecked(ex.getCause());
065        return new ConcurrentException(ex.getMessage(), ex.getCause());
066    }
067
068    /**
069     * Inspects the cause of the specified {@link ExecutionException} and
070     * creates a {@link ConcurrentRuntimeException} with the checked cause if
071     * necessary. This method works exactly like
072     * {@link #extractCause(ExecutionException)}. The only difference is that
073     * the cause of the specified {@link ExecutionException} is extracted as a
074     * runtime exception. This is an alternative for client code that does not
075     * want to deal with checked exceptions.
076     *
077     * @param ex the exception to be processed
078     * @return a {@link ConcurrentRuntimeException} with the checked cause
079     */
080    public static ConcurrentRuntimeException extractCauseUnchecked(
081            final ExecutionException ex) {
082        if (ex == null || ex.getCause() == null) {
083            return null;
084        }
085
086        ExceptionUtils.throwUnchecked(ex.getCause());
087        return new ConcurrentRuntimeException(ex.getMessage(), ex.getCause());
088    }
089
090    /**
091     * Handles the specified {@link ExecutionException}. This method calls
092     * {@link #extractCause(ExecutionException)} for obtaining the cause of the
093     * exception - which might already cause an unchecked exception or an error
094     * being thrown. If the cause is a checked exception however, it is wrapped
095     * in a {@link ConcurrentException}, which is thrown. If the passed in
096     * exception is <b>null</b> or has no cause, the method simply returns
097     * without throwing an exception.
098     *
099     * @param ex the exception to be handled
100     * @throws ConcurrentException if the cause of the {@code
101     * ExecutionException} is a checked exception
102     */
103    public static void handleCause(final ExecutionException ex)
104            throws ConcurrentException {
105        final ConcurrentException cex = extractCause(ex);
106
107        if (cex != null) {
108            throw cex;
109        }
110    }
111
112    /**
113     * Handles the specified {@link ExecutionException} and transforms it into a
114     * runtime exception. This method works exactly like
115     * {@link #handleCause(ExecutionException)}, but instead of a
116     * {@link ConcurrentException} it throws a
117     * {@link ConcurrentRuntimeException}. This is an alternative for client
118     * code that does not want to deal with checked exceptions.
119     *
120     * @param ex the exception to be handled
121     * @throws ConcurrentRuntimeException if the cause of the {@code
122     * ExecutionException} is a checked exception; this exception is then
123     * wrapped in the thrown runtime exception
124     */
125    public static void handleCauseUnchecked(final ExecutionException ex) {
126        final ConcurrentRuntimeException crex = extractCauseUnchecked(ex);
127
128        if (crex != null) {
129            throw crex;
130        }
131    }
132
133    /**
134     * Tests whether the specified {@link Throwable} is a checked exception. If
135     * not, an exception is thrown.
136     *
137     * @param ex the {@link Throwable} to check
138     * @return a flag whether the passed in exception is a checked exception
139     * @throws IllegalArgumentException if the {@link Throwable} is not a
140     * checked exception
141     */
142    static Throwable checkedException(final Throwable ex) {
143        Validate.isTrue(ExceptionUtils.isChecked(ex), "Not a checked exception: " + ex);
144        return ex;
145    }
146
147    /**
148     * Invokes the specified {@link ConcurrentInitializer} and returns the
149     * object produced by the initializer. This method just invokes the {@code
150     * get()} method of the given {@link ConcurrentInitializer}. It is
151     * <b>null</b>-safe: if the argument is <b>null</b>, result is also
152     * <b>null</b>.
153     *
154     * @param <T> the type of the object produced by the initializer
155     * @param initializer the {@link ConcurrentInitializer} to be invoked
156     * @return the object managed by the {@link ConcurrentInitializer}
157     * @throws ConcurrentException if the {@link ConcurrentInitializer} throws
158     * an exception
159     */
160    public static <T> T initialize(final ConcurrentInitializer<T> initializer)
161            throws ConcurrentException {
162        return initializer != null ? initializer.get() : null;
163    }
164
165    /**
166     * Invokes the specified {@link ConcurrentInitializer} and transforms
167     * occurring exceptions to runtime exceptions. This method works like
168     * {@link #initialize(ConcurrentInitializer)}, but if the {@code
169     * ConcurrentInitializer} throws a {@link ConcurrentException}, it is
170     * caught, and the cause is wrapped in a {@link ConcurrentRuntimeException}.
171     * So client code does not have to deal with checked exceptions.
172     *
173     * @param <T> the type of the object produced by the initializer
174     * @param initializer the {@link ConcurrentInitializer} to be invoked
175     * @return the object managed by the {@link ConcurrentInitializer}
176     * @throws ConcurrentRuntimeException if the initializer throws an exception
177     */
178    public static <T> T initializeUnchecked(final ConcurrentInitializer<T> initializer) {
179        try {
180            return initialize(initializer);
181        } catch (final ConcurrentException cex) {
182            throw new ConcurrentRuntimeException(cex.getCause());
183        }
184    }
185
186    /**
187     * Puts a value in the specified {@link ConcurrentMap} if the key is not yet
188     * present. This method works similar to the {@code putIfAbsent()} method of
189     * the {@link ConcurrentMap} interface, but the value returned is different.
190     * Basically, this method is equivalent to the following code fragment:
191     *
192     * <pre>
193     * if (!map.containsKey(key)) {
194     *     map.put(key, value);
195     *     return value;
196     * } else {
197     *     return map.get(key);
198     * }
199     * </pre>
200     *
201     * <p>
202     * except that the action is performed atomically. So this method always
203     * returns the value which is stored in the map.
204     * </p>
205     * <p>
206     * This method is <b>null</b>-safe: It accepts a <b>null</b> map as input
207     * without throwing an exception. In this case the return value is
208     * <b>null</b>, too.
209     * </p>
210     *
211     * @param <K> the type of the keys of the map
212     * @param <V> the type of the values of the map
213     * @param map the map to be modified
214     * @param key the key of the value to be added
215     * @param value the value to be added
216     * @return the value stored in the map after this operation
217     */
218    public static <K, V> V putIfAbsent(final ConcurrentMap<K, V> map, final K key, final V value) {
219        if (map == null) {
220            return null;
221        }
222
223        final V result = map.putIfAbsent(key, value);
224        return result != null ? result : value;
225    }
226
227    /**
228     * Checks if a concurrent map contains a key and creates a corresponding
229     * value if not. This method first checks the presence of the key in the
230     * given map. If it is already contained, its value is returned. Otherwise
231     * the {@code get()} method of the passed in {@link ConcurrentInitializer}
232     * is called. With the resulting object
233     * {@link #putIfAbsent(ConcurrentMap, Object, Object)} is called. This
234     * handles the case that in the meantime another thread has added the key to
235     * the map. Both the map and the initializer can be <b>null</b>; in this
236     * case this method simply returns <b>null</b>.
237     *
238     * @param <K> the type of the keys of the map
239     * @param <V> the type of the values of the map
240     * @param map the map to be modified
241     * @param key the key of the value to be added
242     * @param init the {@link ConcurrentInitializer} for creating the value
243     * @return the value stored in the map after this operation; this may or may
244     * not be the object created by the {@link ConcurrentInitializer}
245     * @throws ConcurrentException if the initializer throws an exception
246     */
247    public static <K, V> V createIfAbsent(final ConcurrentMap<K, V> map, final K key,
248            final ConcurrentInitializer<V> init) throws ConcurrentException {
249        if (map == null || init == null) {
250            return null;
251        }
252
253        final V value = map.get(key);
254        if (value == null) {
255            return putIfAbsent(map, key, init.get());
256        }
257        return value;
258    }
259
260    /**
261     * Checks if a concurrent map contains a key and creates a corresponding
262     * value if not, suppressing checked exceptions. This method calls
263     * {@code createIfAbsent()}. If a {@link ConcurrentException} is thrown, it
264     * is caught and re-thrown as a {@link ConcurrentRuntimeException}.
265     *
266     * @param <K> the type of the keys of the map
267     * @param <V> the type of the values of the map
268     * @param map the map to be modified
269     * @param key the key of the value to be added
270     * @param init the {@link ConcurrentInitializer} for creating the value
271     * @return the value stored in the map after this operation; this may or may
272     * not be the object created by the {@link ConcurrentInitializer}
273     * @throws ConcurrentRuntimeException if the initializer throws an exception
274     */
275    public static <K, V> V createIfAbsentUnchecked(final ConcurrentMap<K, V> map,
276            final K key, final ConcurrentInitializer<V> init) {
277        try {
278            return createIfAbsent(map, key, init);
279        } catch (final ConcurrentException cex) {
280            throw new ConcurrentRuntimeException(cex.getCause());
281        }
282    }
283
284    /**
285     * Gets an implementation of {@link Future} that is immediately done
286     * and returns the specified constant value.
287     *
288     * <p>
289     * This can be useful to return a simple constant immediately from the
290     * concurrent processing, perhaps as part of avoiding nulls.
291     * A constant future can also be useful in testing.
292     * </p>
293     *
294     * @param <T> the type of the value used by this {@link Future} object
295     * @param value  the constant value to return, may be null
296     * @return an instance of Future that will return the value, never null
297     */
298    public static <T> Future<T> constantFuture(final T value) {
299        return new ConstantFuture<>(value);
300    }
301
302    /**
303     * A specialized {@link Future} implementation which wraps a constant value.
304     * @param <T> the type of the value wrapped by this class
305     */
306    static final class ConstantFuture<T> implements Future<T> {
307        /** The constant value. */
308        private final T value;
309
310        /**
311         * Creates a new instance of {@link ConstantFuture} and initializes it
312         * with the constant value.
313         *
314         * @param value the value (may be <b>null</b>)
315         */
316        ConstantFuture(final T value) {
317            this.value = value;
318        }
319
320        /**
321         * {@inheritDoc} This implementation always returns <b>true</b> because
322         * the constant object managed by this {@link Future} implementation is
323         * always available.
324         */
325        @Override
326        public boolean isDone() {
327            return true;
328        }
329
330        /**
331         * {@inheritDoc} This implementation just returns the constant value.
332         */
333        @Override
334        public T get() {
335            return value;
336        }
337
338        /**
339         * {@inheritDoc} This implementation just returns the constant value; it
340         * does not block, therefore the timeout has no meaning.
341         */
342        @Override
343        public T get(final long timeout, final TimeUnit unit) {
344            return value;
345        }
346
347        /**
348         * {@inheritDoc} This implementation always returns <b>false</b>; there
349         * is no background process which could be cancelled.
350         */
351        @Override
352        public boolean isCancelled() {
353            return false;
354        }
355
356        /**
357         * {@inheritDoc} The cancel operation is not supported. This
358         * implementation always returns <b>false</b>.
359         */
360        @Override
361        public boolean cancel(final boolean mayInterruptIfRunning) {
362            return false;
363        }
364    }
365
366}