| /** |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You 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.apache.camel.component.jms.reply; |
| |
| import java.util.concurrent.ExecutorService; |
| import java.util.concurrent.ScheduledExecutorService; |
| |
| import org.apache.camel.support.DefaultTimeoutMap; |
| |
| /** |
| * A {@link org.apache.camel.TimeoutMap} which is used to track reply messages which |
| * has been timed out, and thus should trigger the waiting {@link org.apache.camel.Exchange} to |
| * timeout as well. |
| * |
| * @version |
| */ |
| public class CorrelationTimeoutMap extends DefaultTimeoutMap<String, ReplyHandler> { |
| |
| private CorrelationListener listener; |
| private ExecutorService executorService; |
| |
| public CorrelationTimeoutMap(ScheduledExecutorService executor, long requestMapPollTimeMillis, ExecutorService executorService) { |
| super(executor, requestMapPollTimeMillis); |
| this.executorService = executorService; |
| } |
| |
| public void setListener(CorrelationListener listener) { |
| // there is only one listener needed |
| this.listener = listener; |
| } |
| |
| public boolean onEviction(final String key, final ReplyHandler value) { |
| try { |
| if (listener != null) { |
| listener.onEviction(key); |
| } |
| } catch (Throwable e) { |
| // ignore |
| } |
| |
| final Runnable task = new Runnable() { |
| @Override |
| public void run() { |
| // trigger timeout |
| try { |
| value.onTimeout(key); |
| } catch (Throwable e) { |
| // must ignore so we ensure we evict the element |
| log.warn("Error processing onTimeout for correlationID: " + key + " due: " + e.getMessage() + ". This exception is ignored.", e); |
| } |
| } |
| }; |
| if (executorService != null) { |
| executorService.submit(task); |
| } else { |
| // run task synchronously |
| task.run(); |
| } |
| |
| // return true to remove the element |
| log.trace("Evicted correlationID: {}", key); |
| return true; |
| } |
| |
| @Override |
| public ReplyHandler get(String key) { |
| ReplyHandler answer = super.get(key); |
| log.trace("Get correlationID: {} -> {}", key, answer != null); |
| return answer; |
| } |
| |
| @Override |
| public ReplyHandler put(String key, ReplyHandler value, long timeoutMillis) { |
| try { |
| if (listener != null) { |
| listener.onPut(key); |
| } |
| } catch (Throwable e) { |
| // ignore |
| } |
| |
| ReplyHandler result; |
| if (timeoutMillis <= 0) { |
| // no timeout (must use Integer.MAX_VALUE) |
| result = super.put(key, value, Integer.MAX_VALUE); |
| } else { |
| result = super.put(key, value, timeoutMillis); |
| } |
| log.trace("Added correlationID: {} to timeout after: {} millis", key, timeoutMillis); |
| return result; |
| } |
| |
| @Override |
| public ReplyHandler putIfAbsent(String key, ReplyHandler value, long timeoutMillis) { |
| try { |
| if (listener != null) { |
| listener.onPut(key); |
| } |
| } catch (Throwable e) { |
| // ignore |
| } |
| |
| ReplyHandler result; |
| if (timeoutMillis <= 0) { |
| // no timeout (must use Integer.MAX_VALUE) |
| result = super.putIfAbsent(key, value, Integer.MAX_VALUE); |
| } else { |
| result = super.putIfAbsent(key, value, timeoutMillis); |
| } |
| if (result == null) { |
| log.trace("Added correlationID: {} to timeout after: {} millis", key, timeoutMillis); |
| } else { |
| log.trace("Duplicate correlationID: {} detected", key); |
| } |
| return result; |
| } |
| |
| @Override |
| public ReplyHandler remove(String key) { |
| try { |
| if (listener != null) { |
| listener.onRemove(key); |
| } |
| } catch (Throwable e) { |
| // ignore |
| } |
| |
| ReplyHandler answer = super.remove(key); |
| log.trace("Removed correlationID: {} -> {}", key, answer != null); |
| return answer; |
| } |
| |
| } |