Skip to content

Commit

Permalink
Fix retrying of SocketTimeoutExceptions in HttpConnector
Browse files Browse the repository at this point in the history
As part of investigation of bazelbuild#8974 I found that recent change 982e0b8 broke retries of `SocketTimeoutException` in `HttpConnector`, intention of that change was good but lack of tests resulted in broken logic.

IntelliJ highlights the problem ? branch with `instanceof SocketTimeoutException` would have never been executed:

<img width="764" alt="Screen Shot 2019-07-29 at 3 26 11 PM" src="https://user-images.githubusercontent.com/967132/62089179-d7bfa400-b21c-11e9-882a-1c1fe1fcb683.png">

---

This PR adds missing tests and fixes the logic to still present `SocketTimeoutException` as `IOException` for upstream consumers while handling it properly internally in the `HttpConnector`.

Closes bazelbuild#9008.

PiperOrigin-RevId: 261675244
  • Loading branch information
artem-zinnatullin authored and copybara-github committed Aug 5, 2019
1 parent 299655e commit 338829f
Show file tree
Hide file tree
Showing 4 changed files with 335 additions and 8 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -185,11 +185,6 @@ URLConnection connect(
throw e;
} catch (IllegalArgumentException e) {
throw new UnrecoverableHttpException(e.getMessage());
} catch (SocketTimeoutException e) {
// SocketTimeoutExceptions are InterruptedExceptions; however they do not signify
// an external interruption, but simply a failed download due to some server timing
// out. So rethrow them as ordinary IOExceptions.
throw new IOException(e.getMessage(), e);
} catch (IOException e) {
if (connection != null) {
// If we got here, it means we might not have consumed the entire payload of the
Expand All @@ -214,7 +209,12 @@ URLConnection connect(
throw e;
}
if (++retries == MAX_RETRIES) {
if (!(e instanceof SocketTimeoutException)) {
if (e instanceof SocketTimeoutException) {
// SocketTimeoutExceptions are InterruptedIOExceptions; however they do not signify
// an external interruption, but simply a failed download due to some server timing
// out. So rethrow them as ordinary IOExceptions.
e = new IOException(e.getMessage(), e);
} else {
eventHandler
.handle(Event.progress(format("Error connecting to %s: %s", url, e.getMessage())));
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@
import java.net.Proxy;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketTimeoutException;
import java.net.URL;
import java.net.URLConnection;
import java.util.Locale;
Expand All @@ -53,6 +54,7 @@
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import org.junit.After;
import org.junit.Before;
import org.junit.Rule;
Expand All @@ -79,10 +81,13 @@ public class HttpConnectorTest {
private final ExecutorService executor = Executors.newFixedThreadPool(2);
private final ManualClock clock = new ManualClock();
private final ManualSleeper sleeper = new ManualSleeper(clock);
/** Scale timeouts down to make tests fast. */
private final float timeoutScaling = 0.05f;

private final EventHandler eventHandler = mock(EventHandler.class);
private final ProxyHelper proxyHelper = mock(ProxyHelper.class);
private final HttpConnector connector =
new HttpConnector(Locale.US, eventHandler, proxyHelper, sleeper);
new HttpConnector(Locale.US, eventHandler, proxyHelper, sleeper, timeoutScaling);

@Before
public void before() throws Exception {
Expand Down Expand Up @@ -202,6 +207,159 @@ public Object call() throws Exception {
}
}

@Test
public void connectionRefused_retries() throws Exception {
final int port;

// Start and immediately stop server socket to get a free port.
try (ServerSocket server = new ServerSocket(0, 1, InetAddress.getByName(null))) {
port = server.getLocalPort();
}

final AtomicReference<ServerSocket> server = new AtomicReference<>();

try {
// Schedule server socket to be started only after retry to simulate connection retry.
sleeper.scheduleRunnable(
() -> {
try {
server.set(new ServerSocket(port, 1, InetAddress.getByName(null)));
} catch (IOException e) {
throw new RuntimeException(e);
}

@SuppressWarnings("unused")
Future<?> possiblyIgnoredError =
executor.submit(
() -> {
while (!executor.isShutdown()) {
try (Socket socket = server.get().accept()) {
readHttpRequest(socket.getInputStream());
sendLines(
socket,
"HTTP/1.1 200 OK",
"Date: Fri, 31 Dec 1999 23:59:59 GMT",
"Connection: close",
"Content-Type: text/plain",
"Content-Length: 5",
"",
"hello");
}
}

return null;
});
},
1);

try (Reader payload =
new InputStreamReader(
connector
.connect(
new URL(String.format("http://localhost:%d", port)),
ImmutableMap.<String, String>of())
.getInputStream(),
ISO_8859_1)) {
assertThat(CharStreams.toString(payload)).isEqualTo("hello");
}
} finally {
ServerSocket serverSocket = server.get();

if (serverSocket != null) {
serverSocket.close();
}
}
}

@Test
public void socketTimeout_retries() throws Exception {
try (ServerSocket server = new ServerSocket(0, 1, InetAddress.getByName(null))) {
@SuppressWarnings("unused")
Future<?> possiblyIgnoredError =
executor.submit(
() -> {
try (Socket socket = server.accept()) {
// Do nothing to cause SocketTimeoutException on client side.
}

// Schedule proper HTTP response once client retries.
sleeper.scheduleRunnable(
() -> {
@SuppressWarnings("unused")
Future<?> possiblyIgnoredError2 =
executor.submit(
() -> {
while (!executor.isShutdown()) {
try (Socket socket = server.accept()) {
readHttpRequest(socket.getInputStream());
sendLines(
socket,
"HTTP/1.1 200 OK",
"Date: Fri, 31 Dec 1999 23:59:59 GMT",
"Connection: close",
"Content-Type: text/plain",
"Content-Length: 5",
"",
"hello");
} catch (IOException e) {
throw new RuntimeException(e);
}
}
});
},
1);
return null;
});

try (Reader payload =
new InputStreamReader(
connector
.connect(
new URL(String.format("http://localhost:%d", server.getLocalPort())),
ImmutableMap.<String, String>of())
.getInputStream(),
ISO_8859_1)) {
assertThat(CharStreams.toString(payload)).isEqualTo("hello");
assertThat(clock.currentTimeMillis()).isEqualTo(1);
}
}
}

/**
* It is important part of {@link HttpConnector} contract to not throw raw {@link
* SocketTimeoutException} because it extends {@link java.io.InterruptedIOException} and {@link
* HttpConnectorMultiplexer} relies on {@link java.io.InterruptedIOException} to only be thrown
* when actual interruption happened.
*/
@Test
public void socketTimeout_throwsIOExceptionInsteadOfSocketTimeoutException() throws Exception {
try (ServerSocket server = new ServerSocket(0, 1, InetAddress.getByName(null))) {
@SuppressWarnings("unused")
Future<?> possiblyIgnoredError =
executor.submit(
() -> {
try (Socket socket = server.accept()) {
// Do nothing to cause SocketTimeoutException on client side.
}
return null;
});

try (Reader payload =
new InputStreamReader(
connector
.connect(
new URL(String.format("http://localhost:%d", server.getLocalPort())),
ImmutableMap.<String, String>of())
.getInputStream(),
ISO_8859_1)) {
fail("Should have thrown");
} catch (IOException expected) {
assertThat(expected).hasCauseThat().isInstanceOf(SocketTimeoutException.class);
assertThat(expected).hasCauseThat().hasMessageThat().isEqualTo("connect timed out");
}
}
}

@Test
public void permanentError_doesNotRetryAndThrowsIOException() throws Exception {
try (ServerSocket server = new ServerSocket(0, 1, InetAddress.getByName(null))) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,12 +17,17 @@
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;

import com.google.devtools.build.lib.util.Pair;
import com.google.devtools.build.lib.util.Sleeper;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

/** Fake sleeper for testing. */
public final class ManualSleeper implements Sleeper {

private final ManualClock clock;
private final List<Pair<Long, Runnable>> scheduledRunnables = new ArrayList<>(0);

public ManualSleeper(ManualClock clock) {
this.clock = checkNotNull(clock);
Expand All @@ -31,6 +36,31 @@ public ManualSleeper(ManualClock clock) {
@Override
public void sleepMillis(long milliseconds) throws InterruptedException {
checkArgument(milliseconds >= 0, "sleeper can't time travel");
clock.advanceMillis(milliseconds);
final long resultedCurrentTimeMillis = clock.advanceMillis(milliseconds);

Iterator<Pair<Long, Runnable>> iterator = scheduledRunnables.iterator();

// Run those scheduled Runnables who's time has come.
while (iterator.hasNext()) {
Pair<Long, Runnable> scheduledRunnable = iterator.next();

if (resultedCurrentTimeMillis >= scheduledRunnable.first) {
iterator.remove();
scheduledRunnable.second.run();
}
}
}

/**
* Schedules a given {@link Runnable} to run when this Sleeper's clock has been adjusted with
* {@link #sleepMillis(long)} by {@code delayMilliseconds} or greater.
*
* @param runnable runnable to run, must not throw exceptions.
* @param delayMilliseconds delay in milliseconds from current value of {@link ManualClock} used
* by this {@link ManualSleeper}.
*/
public void scheduleRunnable(Runnable runnable, long delayMilliseconds) {
checkArgument(delayMilliseconds >= 0, "sleeper can't time travel");
scheduledRunnables.add(new Pair<>(clock.currentTimeMillis() + delayMilliseconds, runnable));
}
}
Loading

0 comments on commit 338829f

Please sign in to comment.