ARTEMIS-2691 Improve critical analyzer LOG policy

Reset the critical component timer after the expiration to allow further LOG
actions after the first expiration.
This commit is contained in:
brusdev 2020-04-03 16:31:08 +02:00 committed by Clebert Suconic
parent dd4b89d881
commit 38a79be314
8 changed files with 76 additions and 38 deletions

View File

@ -136,8 +136,8 @@ public class CriticalAnalyzerImpl implements CriticalAnalyzer {
try { try {
for (CriticalComponent component : components) { for (CriticalComponent component : components) {
if (component.isExpired(timeoutNanoSeconds)) { if (component.checkExpiration(timeoutNanoSeconds, true)) {
fireAction(component); fireActions(component);
// no need to keep running if there's already a component failed // no need to keep running if there's already a component failed
return; return;
} }
@ -149,7 +149,7 @@ public class CriticalAnalyzerImpl implements CriticalAnalyzer {
} }
} }
private void fireAction(CriticalComponent component) { private void fireActions(CriticalComponent component) {
for (CriticalAction action : actions) { for (CriticalAction action : actions) {
try { try {
action.run(component); action.run(component);
@ -157,14 +157,11 @@ public class CriticalAnalyzerImpl implements CriticalAnalyzer {
logger.warn(e.getMessage(), e); logger.warn(e.getMessage(), e);
} }
} }
actions.clear();
} }
@Override @Override
public void start() { public void start() {
scheduledComponent.start(); scheduledComponent.start();
} }
@Override @Override

View File

@ -49,9 +49,10 @@ public interface CriticalComponent {
} }
/** /**
* Is this Component expired at a given timeout.. on any of its paths. * Check if the component is expired at a given timeout.. on any of its paths.
* @param timeout * @param timeout - the timeout to check if the component is expired
* @param reset - true to reset the component timer if it is expired
* @return -1 if it's ok, or the number of the path it failed * @return -1 if it's ok, or the number of the path it failed
*/ */
boolean isExpired(long timeout); boolean checkExpiration(long timeout, boolean reset);
} }

View File

@ -62,9 +62,9 @@ public class CriticalComponentImpl implements CriticalComponent {
} }
@Override @Override
public boolean isExpired(long timeout) { public boolean checkExpiration(long timeout, boolean reset) {
for (int i = 0; i < measures.length; i++) { for (int i = 0; i < measures.length; i++) {
if (measures[i].isExpired(timeout)) { if (measures[i].checkExpiration(timeout, reset)) {
return true; return true;
} }
} }

View File

@ -64,7 +64,7 @@ public class CriticalMeasure {
CriticalAnalyzer analyzer = component != null ? component.getCriticalAnalyzer() : null; CriticalAnalyzer analyzer = component != null ? component.getCriticalAnalyzer() : null;
if (analyzer != null) { if (analyzer != null) {
long nanoTimeout = analyzer.getTimeoutNanoSeconds(); long nanoTimeout = analyzer.getTimeoutNanoSeconds();
if (isExpired(nanoTimeout)) { if (checkExpiration(nanoTimeout, false)) {
logger.trace("Path " + id + " on component " + getComponentName() + " is taking too long, leaving at", new Exception("entered")); logger.trace("Path " + id + " on component " + getComponentName() + " is taking too long, leaving at", new Exception("entered"));
logger.trace("Path " + id + " on component " + getComponentName() + " is taking too long, entered at", traceEnter); logger.trace("Path " + id + " on component " + getComponentName() + " is taking too long, entered at", traceEnter);
} }
@ -83,7 +83,8 @@ public class CriticalMeasure {
} }
} }
public boolean isExpired(long timeout) { public boolean checkExpiration(long timeout, boolean reset) {
long time = System.nanoTime();
final long timeLeft = TIME_LEFT_UPDATER.get(this); final long timeLeft = TIME_LEFT_UPDATER.get(this);
final long timeEnter = TIME_ENTER_UPDATER.get(this); final long timeEnter = TIME_ENTER_UPDATER.get(this);
//due to how System::nanoTime works is better to use differences to prevent numerical overflow while comparing //due to how System::nanoTime works is better to use differences to prevent numerical overflow while comparing
@ -91,13 +92,18 @@ public class CriticalMeasure {
boolean expired = System.nanoTime() - timeEnter > timeout; boolean expired = System.nanoTime() - timeEnter > timeout;
if (expired) { if (expired) {
Exception lastTraceEnter = this.traceEnter;
Exception thistraceEnter = this.traceEnter; if (lastTraceEnter != null) {
if (thistraceEnter != null) { logger.warn("Component " + getComponentName() + " is expired on path " + id, lastTraceEnter);
logger.warn("Component " + getComponentName() + " is expired on path " + id, thistraceEnter);
} else { } else {
logger.warn("Component " + getComponentName() + " is expired on path " + id); logger.warn("Component " + getComponentName() + " is expired on path " + id);
} }
if (reset) {
TIME_LEFT_UPDATER.lazySet(this, time);
TIME_ENTER_UPDATER.lazySet(this, time);
}
} }
return expired; return expired;
} }

View File

@ -20,6 +20,7 @@ package org.apache.activemq.artemis.utils.critical;
import java.util.concurrent.CountDownLatch; import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import org.apache.activemq.artemis.utils.ReusableLatch;
import org.apache.activemq.artemis.utils.ThreadLeakCheckRule; import org.apache.activemq.artemis.utils.ThreadLeakCheckRule;
import org.junit.After; import org.junit.After;
import org.junit.Assert; import org.junit.Assert;
@ -45,7 +46,7 @@ public class CriticalAnalyzerTest {
analyzer = new CriticalAnalyzerImpl().setTimeout(100, TimeUnit.MILLISECONDS).setCheckTime(50, TimeUnit.MILLISECONDS); analyzer = new CriticalAnalyzerImpl().setTimeout(100, TimeUnit.MILLISECONDS).setCheckTime(50, TimeUnit.MILLISECONDS);
analyzer.add(new CriticalComponent() { analyzer.add(new CriticalComponent() {
@Override @Override
public boolean isExpired(long timeout) { public boolean checkExpiration(long timeout, boolean reset) {
return true; return true;
} }
}); });
@ -66,7 +67,15 @@ public class CriticalAnalyzerTest {
@Test @Test
public void testActionOnImpl() throws Exception { public void testActionOnImpl() throws Exception {
CountDownLatch latch = new CountDownLatch(1);
analyzer = new CriticalAnalyzerImpl().setTimeout(10, TimeUnit.MILLISECONDS).setCheckTime(5, TimeUnit.MILLISECONDS); analyzer = new CriticalAnalyzerImpl().setTimeout(10, TimeUnit.MILLISECONDS).setCheckTime(5, TimeUnit.MILLISECONDS);
analyzer.addAction((CriticalComponent comp) -> {
System.out.println("component " + comp + " received");
latch.countDown();
});
CriticalComponent component = new CriticalComponentImpl(analyzer, 2); CriticalComponent component = new CriticalComponentImpl(analyzer, 2);
analyzer.add(component); analyzer.add(component);
@ -74,15 +83,9 @@ public class CriticalAnalyzerTest {
component.leaveCritical(0); component.leaveCritical(0);
component.enterCritical(1); component.enterCritical(1);
CountDownLatch latch = new CountDownLatch(1);
analyzer.start(); analyzer.start();
analyzer.addAction((CriticalComponent comp) -> {
System.out.println("component " + comp + " received");
latch.countDown();
});
Assert.assertTrue(latch.await(10, TimeUnit.SECONDS)); Assert.assertTrue(latch.await(10, TimeUnit.SECONDS));
analyzer.stop(); analyzer.stop();
@ -93,7 +96,7 @@ public class CriticalAnalyzerTest {
analyzer = new CriticalAnalyzerImpl().setTimeout(10, TimeUnit.MILLISECONDS).setCheckTime(5, TimeUnit.MILLISECONDS); analyzer = new CriticalAnalyzerImpl().setTimeout(10, TimeUnit.MILLISECONDS).setCheckTime(5, TimeUnit.MILLISECONDS);
CriticalComponent component = new CriticalComponentImpl(analyzer, 2); CriticalComponent component = new CriticalComponentImpl(analyzer, 2);
component.enterCritical(0); component.enterCritical(0);
Assert.assertFalse(component.isExpired(TimeUnit.MINUTES.toNanos(1))); Assert.assertFalse(component.checkExpiration(TimeUnit.MINUTES.toNanos(1), false));
analyzer.stop(); analyzer.stop();
} }
@ -104,32 +107,62 @@ public class CriticalAnalyzerTest {
CriticalComponent component = new CriticalComponentImpl(analyzer, 2); CriticalComponent component = new CriticalComponentImpl(analyzer, 2);
component.enterCritical(0); component.enterCritical(0);
Thread.sleep(50); Thread.sleep(50);
Assert.assertTrue(component.isExpired(0)); Assert.assertTrue(component.checkExpiration(0, false));
analyzer.stop(); analyzer.stop();
} }
@Test @Test
public void testNegative() throws Exception { public void testNegative() throws Exception {
analyzer = new CriticalAnalyzerImpl().setTimeout(10, TimeUnit.MILLISECONDS).setCheckTime(5, TimeUnit.MILLISECONDS);
CriticalComponent component = new CriticalComponentImpl(analyzer, 1);
analyzer.add(component);
component.enterCritical(0);
component.leaveCritical(0);
CountDownLatch latch = new CountDownLatch(1); CountDownLatch latch = new CountDownLatch(1);
analyzer.start(); analyzer = new CriticalAnalyzerImpl().setTimeout(10, TimeUnit.MILLISECONDS).setCheckTime(5, TimeUnit.MILLISECONDS);
analyzer.addAction((CriticalComponent comp) -> { analyzer.addAction((CriticalComponent comp) -> {
System.out.println("component " + comp + " received"); System.out.println("component " + comp + " received");
latch.countDown(); latch.countDown();
}); });
CriticalComponent component = new CriticalComponentImpl(analyzer, 1);
analyzer.add(component);
component.enterCritical(0);
component.leaveCritical(0);
analyzer.start();
Assert.assertFalse(latch.await(100, TimeUnit.MILLISECONDS)); Assert.assertFalse(latch.await(100, TimeUnit.MILLISECONDS));
analyzer.stop(); analyzer.stop();
} }
@Test
public void testPositive() throws Exception {
ReusableLatch latch = new ReusableLatch(1);
analyzer = new CriticalAnalyzerImpl().setTimeout(10, TimeUnit.MILLISECONDS).setCheckTime(5, TimeUnit.MILLISECONDS);
analyzer.addAction((CriticalComponent comp) -> {
System.out.println("component " + comp + " received");
latch.countDown();
});
CriticalComponent component = new CriticalComponentImpl(analyzer, 1);
analyzer.add(component);
component.enterCritical(0);
Thread.sleep(50);
analyzer.start();
Assert.assertTrue(latch.await(100, TimeUnit.MILLISECONDS));
component.leaveCritical(0);
latch.setCount(1);
Assert.assertFalse(latch.await(100, TimeUnit.MILLISECONDS));
analyzer.stop();
}
} }

View File

@ -30,7 +30,7 @@ public class CriticalMeasureTest {
long time = System.nanoTime(); long time = System.nanoTime();
CriticalMeasure.TIME_ENTER_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(5)); CriticalMeasure.TIME_ENTER_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(5));
CriticalMeasure.TIME_LEFT_UPDATER.set(measure, time); CriticalMeasure.TIME_LEFT_UPDATER.set(measure, time);
Assert.assertFalse(measure.isExpired(TimeUnit.SECONDS.toNanos(30))); Assert.assertFalse(measure.checkExpiration(TimeUnit.SECONDS.toNanos(30), false));
} }
@Test @Test
@ -41,7 +41,7 @@ public class CriticalMeasureTest {
long time = System.nanoTime(); long time = System.nanoTime();
CriticalMeasure.TIME_ENTER_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(5)); CriticalMeasure.TIME_ENTER_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(5));
measure.leaveCritical(); measure.leaveCritical();
Assert.assertFalse(measure.isExpired(TimeUnit.SECONDS.toNanos(30))); Assert.assertFalse(measure.checkExpiration(TimeUnit.SECONDS.toNanos(30), false));
} }
@Test @Test
@ -53,7 +53,7 @@ public class CriticalMeasureTest {
measure.enterCritical(); measure.enterCritical();
CriticalMeasure.TIME_ENTER_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(5)); CriticalMeasure.TIME_ENTER_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(5));
CriticalMeasure.TIME_LEFT_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(10)); CriticalMeasure.TIME_LEFT_UPDATER.set(measure, time - TimeUnit.MINUTES.toNanos(10));
Assert.assertTrue(measure.isExpired(TimeUnit.SECONDS.toNanos(30))); Assert.assertTrue(measure.checkExpiration(TimeUnit.SECONDS.toNanos(30), false));
measure.leaveCritical(); measure.leaveCritical();
} }
} }

View File

@ -1265,6 +1265,7 @@ public class ActiveMQServerImpl implements ActiveMQServer {
} }
try { try {
this.analyzer.clear();
this.analyzer.stop(); this.analyzer.stop();
} catch (Exception e) { } catch (Exception e) {
logger.warn(e.getMessage(), e); logger.warn(e.getMessage(), e);

View File

@ -54,7 +54,7 @@ public class CriticalSimpleTest extends ActiveMQTestBase {
server.getCriticalAnalyzer().add(new CriticalComponent() { server.getCriticalAnalyzer().add(new CriticalComponent() {
@Override @Override
public boolean isExpired(long timeout) { public boolean checkExpiration(long timeout, boolean reset) {
return true; return true;
} }
}); });
@ -83,7 +83,7 @@ public class CriticalSimpleTest extends ActiveMQTestBase {
try { try {
server.getCriticalAnalyzer().add(new CriticalComponent() { server.getCriticalAnalyzer().add(new CriticalComponent() {
@Override @Override
public boolean isExpired(long timeout) { public boolean checkExpiration(long timeout, boolean reset) {
return true; return true;
} }
}); });