Handle timeouts better in lock handler

Rather than just logging a generic TimeoutException, this will say what action
timed out and how long it had been executing for.

-------------
Created by MOE: https://github.com/google/moe
MOE_MIGRATED_REVID=201049025
This commit is contained in:
mcilwain 2018-06-18 13:32:02 -07:00 committed by Ben McIlwain
parent 7a4a4ba3d5
commit f971583dc0
2 changed files with 53 additions and 11 deletions

View file

@ -18,12 +18,14 @@ import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Throwables.throwIfUnchecked; import static com.google.common.base.Throwables.throwIfUnchecked;
import com.google.common.annotations.VisibleForTesting; import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Joiner;
import com.google.common.base.Strings; import com.google.common.base.Strings;
import com.google.common.collect.ImmutableSortedSet; import com.google.common.collect.ImmutableSortedSet;
import com.google.common.flogger.FluentLogger; import com.google.common.flogger.FluentLogger;
import com.google.common.util.concurrent.UncheckedExecutionException; import com.google.common.util.concurrent.UncheckedExecutionException;
import google.registry.model.server.Lock; import google.registry.model.server.Lock;
import google.registry.util.AppEngineTimeLimiter; import google.registry.util.AppEngineTimeLimiter;
import google.registry.util.Clock;
import google.registry.util.RequestStatusChecker; import google.registry.util.RequestStatusChecker;
import java.util.HashSet; import java.util.HashSet;
import java.util.Optional; import java.util.Optional;
@ -31,8 +33,10 @@ import java.util.Set;
import java.util.concurrent.Callable; import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import javax.annotation.Nullable; import javax.annotation.Nullable;
import javax.inject.Inject; import javax.inject.Inject;
import org.joda.time.DateTime;
import org.joda.time.Duration; import org.joda.time.Duration;
/** Implementation of {@link LockHandler} that uses the datastore lock. */ /** Implementation of {@link LockHandler} that uses the datastore lock. */
@ -45,6 +49,7 @@ public class LockHandlerImpl implements LockHandler {
/** Fudge factor to make sure we kill threads before a lock actually expires. */ /** Fudge factor to make sure we kill threads before a lock actually expires. */
private static final Duration LOCK_TIMEOUT_FUDGE = Duration.standardSeconds(5); private static final Duration LOCK_TIMEOUT_FUDGE = Duration.standardSeconds(5);
@Inject Clock clock;
@Inject RequestStatusChecker requestStatusChecker; @Inject RequestStatusChecker requestStatusChecker;
@Inject public LockHandlerImpl() {} @Inject public LockHandlerImpl() {}
@ -65,14 +70,27 @@ public class LockHandlerImpl implements LockHandler {
@Nullable String tld, @Nullable String tld,
Duration leaseLength, Duration leaseLength,
String... lockNames) { String... lockNames) {
DateTime startTime = clock.nowUtc();
String sanitizedTld = Strings.emptyToNull(tld);
try { try {
return AppEngineTimeLimiter.create().callWithTimeout( return AppEngineTimeLimiter.create()
new LockingCallable(callable, Strings.emptyToNull(tld), leaseLength, lockNames), .callWithTimeout(
leaseLength.minus(LOCK_TIMEOUT_FUDGE).getMillis(), new LockingCallable(callable, sanitizedTld, leaseLength, lockNames),
TimeUnit.MILLISECONDS); leaseLength.minus(LOCK_TIMEOUT_FUDGE).getMillis(),
TimeUnit.MILLISECONDS);
} catch (ExecutionException | UncheckedExecutionException e) { } catch (ExecutionException | UncheckedExecutionException e) {
// Unwrap the execution exception and throw its root cause. // Unwrap the execution exception and throw its root cause.
Throwable cause = e.getCause(); Throwable cause = e.getCause();
if (cause instanceof TimeoutException) {
throw new RuntimeException(
String.format(
"Execution on locks '%s' for TLD '%s' timed out after %s; started at %s",
Joiner.on(", ").join(lockNames),
Optional.ofNullable(sanitizedTld).orElse("(null)"),
new Duration(startTime, clock.nowUtc()),
startTime),
cause);
}
throwIfUnchecked(cause); throwIfUnchecked(cause);
throw new RuntimeException(cause); throw new RuntimeException(cause);
} catch (Exception e) { } catch (Exception e) {

View file

@ -22,9 +22,12 @@ import static org.mockito.Mockito.verify;
import google.registry.model.server.Lock; import google.registry.model.server.Lock;
import google.registry.testing.AppEngineRule; import google.registry.testing.AppEngineRule;
import google.registry.testing.FakeClock;
import java.util.Optional; import java.util.Optional;
import java.util.concurrent.Callable; import java.util.concurrent.Callable;
import java.util.concurrent.TimeoutException;
import javax.annotation.Nullable; import javax.annotation.Nullable;
import org.joda.time.DateTime;
import org.joda.time.Duration; import org.joda.time.Duration;
import org.junit.Before; import org.junit.Before;
import org.junit.Rule; import org.junit.Rule;
@ -38,9 +41,9 @@ public final class LockHandlerImplTest {
private static final Duration ONE_DAY = Duration.standardDays(1); private static final Duration ONE_DAY = Duration.standardDays(1);
@Rule private final FakeClock clock = new FakeClock(DateTime.parse("2001-08-29T12:20:00Z"));
public final AppEngineRule appEngine = AppEngineRule.builder()
.build(); @Rule public final AppEngineRule appEngine = AppEngineRule.builder().build();
private static class CountingCallable implements Callable<Void> { private static class CountingCallable implements Callable<Void> {
int numCalled = 0; int numCalled = 0;
@ -54,19 +57,22 @@ public final class LockHandlerImplTest {
private static class ThrowingCallable implements Callable<Void> { private static class ThrowingCallable implements Callable<Void> {
Exception exception; Exception exception;
FakeClock clock;
ThrowingCallable(Exception exception) { ThrowingCallable(Exception exception, FakeClock clock) {
this.exception = exception; this.exception = exception;
this.clock = clock;
} }
@Override @Override
public Void call() throws Exception { public Void call() throws Exception {
clock.advanceBy(Duration.standardSeconds(77));
throw exception; throw exception;
} }
} }
private boolean executeWithLocks(Callable<Void> callable, final @Nullable Lock acquiredLock) { private boolean executeWithLocks(Callable<Void> callable, final @Nullable Lock acquiredLock) {
LockHandler lockHandler = new LockHandlerImpl() { LockHandlerImpl lockHandler = new LockHandlerImpl() {
private static final long serialVersionUID = 0L; private static final long serialVersionUID = 0L;
@Override @Override
Optional<Lock> acquire(String resourceName, String tld, Duration leaseLength) { Optional<Lock> acquire(String resourceName, String tld, Duration leaseLength) {
@ -76,6 +82,7 @@ public final class LockHandlerImplTest {
return Optional.ofNullable(acquiredLock); return Optional.ofNullable(acquiredLock);
} }
}; };
lockHandler.clock = clock;
return lockHandler.executeWithLocks(callable, "tld", ONE_DAY, "resourceName"); return lockHandler.executeWithLocks(callable, "tld", ONE_DAY, "resourceName");
} }
@ -99,11 +106,28 @@ public final class LockHandlerImplTest {
RuntimeException exception = RuntimeException exception =
assertThrows( assertThrows(
RuntimeException.class, RuntimeException.class,
() -> executeWithLocks(new ThrowingCallable(expectedException), lock)); () -> executeWithLocks(new ThrowingCallable(expectedException, clock), lock));
assertThat(exception).isSameAs(expectedException); assertThat(exception).isSameAs(expectedException);
verify(lock, times(1)).release(); verify(lock, times(1)).release();
} }
@Test
public void testLockSucceeds_timeoutException() {
Lock lock = mock(Lock.class);
Exception expectedException = new TimeoutException("test");
RuntimeException thrown =
assertThrows(
RuntimeException.class,
() -> executeWithLocks(new ThrowingCallable(expectedException, clock), lock));
assertThat(thrown).hasCauseThat().isSameAs(expectedException);
assertThat(thrown)
.hasMessageThat()
.isEqualTo(
"Execution on locks 'resourceName' for TLD 'tld'"
+ " timed out after PT77S; started at 2001-08-29T12:20:00.000Z");
verify(lock, times(1)).release();
}
@Test @Test
public void testLockSucceeds_checkedException() throws Exception { public void testLockSucceeds_checkedException() throws Exception {
Lock lock = mock(Lock.class); Lock lock = mock(Lock.class);
@ -111,7 +135,7 @@ public final class LockHandlerImplTest {
RuntimeException exception = RuntimeException exception =
assertThrows( assertThrows(
RuntimeException.class, RuntimeException.class,
() -> executeWithLocks(new ThrowingCallable(expectedException), lock)); () -> executeWithLocks(new ThrowingCallable(expectedException, clock), lock));
assertThat(exception).hasCauseThat().isSameAs(expectedException); assertThat(exception).hasCauseThat().isSameAs(expectedException);
verify(lock, times(1)).release(); verify(lock, times(1)).release();
} }