You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

LoggingScheduledExecutorService.java 4.0KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. /*
  2. * Copyright (c) 2006-2015 DMDirc Developers
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a copy
  5. * of this software and associated documentation files (the "Software"), to deal
  6. * in the Software without restriction, including without limitation the rights
  7. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. * copies of the Software, and to permit persons to whom the Software is
  9. * furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  19. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  20. * SOFTWARE.
  21. */
  22. package com.dmdirc.util;
  23. import com.google.common.util.concurrent.ThreadFactoryBuilder;
  24. import java.util.concurrent.CancellationException;
  25. import java.util.concurrent.ExecutionException;
  26. import java.util.concurrent.Future;
  27. import java.util.concurrent.ScheduledThreadPoolExecutor;
  28. import java.util.function.BiConsumer;
  29. import org.slf4j.Logger;
  30. import org.slf4j.LoggerFactory;
  31. import static com.dmdirc.util.LogUtils.APP_ERROR;
  32. /**
  33. * An scheduled executor service that takes logs failed executions either via eventbus errors or a
  34. * custom error function.
  35. */
  36. public class LoggingScheduledExecutorService extends ScheduledThreadPoolExecutor {
  37. private static final Logger LOG = LoggerFactory.getLogger(LoggingScheduledExecutorService.class);
  38. private final BiConsumer<Runnable, Throwable> afterExecute;
  39. /**
  40. * Creates a new instance of this executor service.
  41. *
  42. * @param coreSize The number of threads to keep in the pool, even if they are idle, unless
  43. * {@code allowCoreThreadTimeOut} is set
  44. * @param poolName The naming format to use when naming threads
  45. */
  46. public LoggingScheduledExecutorService(final int coreSize, final String poolName) {
  47. this(coreSize, (r, t) -> LOG.error(APP_ERROR, t.getMessage(), t), poolName);
  48. }
  49. /**
  50. * Creates a new instance of this executor service.
  51. *
  52. * @param coreSize The number of threads to keep in the pool, even if they are idle,
  53. * unless {@code allowCoreThreadTimeOut} is set
  54. * @param afterExecute The function to call when an exception occurs
  55. * @param poolName The naming format to use when naming threads
  56. */
  57. public LoggingScheduledExecutorService(final int coreSize,
  58. final BiConsumer<Runnable, Throwable> afterExecute, final String poolName) {
  59. super(coreSize, new ThreadFactoryBuilder().setNameFormat(poolName + "-%d").build());
  60. this.afterExecute = afterExecute;
  61. }
  62. @Override
  63. protected void afterExecute(final Runnable r, final Throwable t) {
  64. super.afterExecute(r, t);
  65. if (t == null && r instanceof Future<?>) {
  66. try {
  67. if (((Future<?>) r).isDone()) {
  68. // If a periodic task completes successfully it is reset before this method
  69. // is called, and get() will block indefinitely. If it is not scheduled to run
  70. // again, or if it has thrown an exception and been stopped by the executor,
  71. // then isDone() will return true and we can safely call get().
  72. ((Future<?>) r).get();
  73. }
  74. } catch (CancellationException | InterruptedException ex) {
  75. //Ignore
  76. } catch (ExecutionException ex) {
  77. afterExecute.accept(r, ex);
  78. }
  79. }
  80. if (t != null) {
  81. afterExecute.accept(r, t);
  82. }
  83. }
  84. }