/* * Copyright (c) MuleSoft, Inc. All rights reserved. http://www.mulesoft.com * The software in this package is published under the terms of the CPAL v1.0 * license, a copy of which has been included with this distribution in the * LICENSE.txt file. */ package org.mule.runtime.core.management.stats; import static java.lang.Thread.currentThread; import static java.util.concurrent.TimeUnit.MILLISECONDS; import org.mule.runtime.api.exception.MuleException; import org.mule.runtime.api.scheduler.Scheduler; import org.mule.runtime.core.api.MuleContext; import org.mule.runtime.core.api.context.MuleContextAware; import java.lang.ref.ReferenceQueue; import java.lang.ref.WeakReference; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.atomic.AtomicLong; public class DefaultProcessingTimeWatcher implements ProcessingTimeWatcher, MuleContextAware { private final ReferenceQueue<ProcessingTime> queue = new ReferenceQueue<ProcessingTime>(); private final Map<ProcessingTimeReference, Object> refs = new ConcurrentHashMap<ProcessingTimeReference, Object>(); private MuleContext muleContext; private Scheduler scheduler; @Override public void addProcessingTime(ProcessingTime processingTime) { refs.put(new ProcessingTimeReference(processingTime, queue), refs); } @Override public void start() throws MuleException { scheduler = muleContext.getSchedulerService().customScheduler(muleContext.getSchedulerBaseConfig() .withName("processing.time.monitor").withMaxConcurrentTasks(1).withShutdownTimeout(0, MILLISECONDS)); scheduler.submit(new ProcessingTimeChecker()); } @Override public void stop() throws MuleException { scheduler.stop(); refs.clear(); } @Override public void setMuleContext(MuleContext muleContext) { this.muleContext = muleContext; } private class ProcessingTimeChecker implements Runnable { /** * As weak references to completed ProcessingTimes are delivered, record them */ @Override public void run() { try { ProcessingTimeReference ref = (ProcessingTimeReference) queue.remove(); refs.remove(ref); FlowConstructStatistics stats = ref.getStatistics(); if (stats.isEnabled()) { stats.addCompleteFlowExecutionTime(ref.getAccumulator().longValue()); } } catch (InterruptedException ex) { currentThread().interrupt(); } } } /** * Weak reference that includes flow statistics to be updated */ static class ProcessingTimeReference extends WeakReference<ProcessingTime> { private FlowConstructStatistics statistics; private AtomicLong accumulator; ProcessingTimeReference(ProcessingTime time, ReferenceQueue<ProcessingTime> queue) { super(time, queue); this.statistics = time.getStatistics(); this.accumulator = time.getAccumulator(); } public AtomicLong getAccumulator() { return accumulator; } public FlowConstructStatistics getStatistics() { return statistics; } } }