From b57d060408bcc58b2afbce671b2e68a07e90d7d9 Mon Sep 17 00:00:00 2001 From: Shawn Pearce Date: Mon, 4 Jul 2016 22:10:55 -0700 Subject: [PATCH] push: Report fatal server errors during pack writing If the push client has requested side-band support the server can signal a fatal error parsing the pack using the error channel (3) and then hang up. This may cause the PackWriter to fail to write to data onto the network socket, which throws a misleading error back up to the application and the user. During a write failure poll the input to see if the side band system can parse out an error message off channel 3. This should be fast as there will either be an error present in the buffer, or the remote will also have hung-up on the side band channel. In the case of a hang-up just rethrow the original IOException as its a network error. This roughly matches what C git does; once commands are sent and the packer is started a new thread runs in the background to decode any possible server error during unpacking on the remote peer Change-Id: Idb37a4a122a565ec4b59130e08c27d963ba09395 --- .../transport/BasePackPushConnection.java | 52 ++++++++++++++++++- 1 file changed, 51 insertions(+), 1 deletion(-) diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/transport/BasePackPushConnection.java b/org.eclipse.jgit/src/org/eclipse/jgit/transport/BasePackPushConnection.java index 9b8ba8098f..0cbbdc77e3 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/transport/BasePackPushConnection.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/transport/BasePackPushConnection.java @@ -47,6 +47,7 @@ package org.eclipse.jgit.transport; import static org.eclipse.jgit.transport.GitProtocolConstants.CAPABILITY_ATOMIC; import java.io.IOException; +import java.io.InputStream; import java.io.OutputStream; import java.text.MessageFormat; import java.util.Collection; @@ -317,7 +318,12 @@ public abstract class BasePackPushConnection extends BasePackConnection implemen writer.setReuseValidatingObjects(false); writer.setDeltaBaseAsOffset(capableOfsDelta); writer.preparePack(monitor, newObjects, remoteObjects); - writer.writePack(monitor, monitor, out); + + OutputStream packOut = out; + if (capableSideBand) { + packOut = new CheckingSideBandOutputStream(in, out); + } + writer.writePack(monitor, monitor, packOut); packTransferTime = writer.getStatistics().getTimeWriting(); } @@ -397,4 +403,48 @@ public abstract class BasePackPushConnection extends BasePackConnection implemen timeoutIn.setTimeout(oldTimeout); } } + + private static class CheckingSideBandOutputStream extends OutputStream { + private final InputStream in; + private final OutputStream out; + + CheckingSideBandOutputStream(InputStream in, OutputStream out) { + this.in = in; + this.out = out; + } + + @Override + public void write(int b) throws IOException { + write(new byte[] { (byte) b }); + } + + @Override + public void write(byte[] buf, int ptr, int cnt) throws IOException { + try { + out.write(buf, ptr, cnt); + } catch (IOException e) { + throw checkError(e); + } + } + + @Override + public void flush() throws IOException { + try { + out.flush(); + } catch (IOException e) { + throw checkError(e); + } + } + + private IOException checkError(IOException e1) { + try { + in.read(); + } catch (TransportException e2) { + return e2; + } catch (IOException e2) { + return e1; + } + return e1; + } + } } -- 2.39.5