Disambiguate pkt-line "0000" from "0004"
[egit/graphgui.git] / org.spearce.jgit / src / org / spearce / jgit / transport / PacketLineIn.java
blob92c700973d274d5da4cfa55e53d5f3c51de6976f
1 /*
2 * Copyright (C) 2008, Robin Rosenberg <robin.rosenberg@dewire.com>
3 * Copyright (C) 2008, Shawn O. Pearce <spearce@spearce.org>
5 * All rights reserved.
7 * Redistribution and use in source and binary forms, with or
8 * without modification, are permitted provided that the following
9 * conditions are met:
11 * - Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
14 * - Redistributions in binary form must reproduce the above
15 * copyright notice, this list of conditions and the following
16 * disclaimer in the documentation and/or other materials provided
17 * with the distribution.
19 * - Neither the name of the Git Development Community nor the
20 * names of its contributors may be used to endorse or promote
21 * products derived from this software without specific prior
22 * written permission.
24 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
25 * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
26 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
27 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
28 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
29 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
31 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
32 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
33 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
34 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
35 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
36 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
39 package org.spearce.jgit.transport;
41 import java.io.IOException;
42 import java.io.InputStream;
43 import java.util.Arrays;
45 import org.spearce.jgit.errors.PackProtocolException;
46 import org.spearce.jgit.lib.Constants;
47 import org.spearce.jgit.lib.MutableObjectId;
48 import org.spearce.jgit.lib.ProgressMonitor;
49 import org.spearce.jgit.util.NB;
50 import org.spearce.jgit.util.RawParseUtils;
52 class PacketLineIn {
53 static final String END = new String("") /* must not string pool */;
54 private static final byte fromhex[];
56 static {
57 fromhex = new byte['f' + 1];
58 Arrays.fill(fromhex, (byte) -1);
59 for (char i = '0'; i <= '9'; i++)
60 fromhex[i] = (byte) (i - '0');
61 for (char i = 'a'; i <= 'f'; i++)
62 fromhex[i] = (byte) ((i - 'a') + 10);
65 static enum AckNackResult {
66 /** NAK */
67 NAK,
68 /** ACK */
69 ACK,
70 /** ACK + continue */
71 ACK_CONTINUE
74 private final InputStream in;
76 private final byte[] lenbuffer;
78 PacketLineIn(final InputStream i) {
79 in = i;
80 lenbuffer = new byte[4];
83 InputStream sideband(final ProgressMonitor pm) {
84 return new SideBandInputStream(this, in, pm);
87 AckNackResult readACK(final MutableObjectId returnedId) throws IOException {
88 final String line = readString();
89 if (line.length() == 0)
90 throw new PackProtocolException("Expected ACK/NAK, found EOF");
91 if ("NAK".equals(line))
92 return AckNackResult.NAK;
93 if (line.startsWith("ACK ")) {
94 returnedId.fromString(line.substring(4, 44));
95 if (line.indexOf("continue", 44) != -1)
96 return AckNackResult.ACK_CONTINUE;
97 return AckNackResult.ACK;
99 throw new PackProtocolException("Expected ACK/NAK, got: " + line);
102 String readString() throws IOException {
103 int len = readLength();
104 if (len == 0)
105 return END;
107 len -= 4; // length header (4 bytes)
108 if (len == 0)
109 return "";
111 final byte[] raw = new byte[len];
112 NB.readFully(in, raw, 0, len);
113 if (raw[len - 1] == '\n')
114 len--;
115 return RawParseUtils.decode(Constants.CHARSET, raw, 0, len);
118 String readStringRaw() throws IOException {
119 int len = readLength();
120 if (len == 0)
121 return END;
123 len -= 4; // length header (4 bytes)
125 final byte[] raw = new byte[len];
126 NB.readFully(in, raw, 0, len);
127 return RawParseUtils.decode(Constants.CHARSET, raw, 0, len);
131 int readLength() throws IOException {
132 NB.readFully(in, lenbuffer, 0, 4);
133 try {
134 int r = fromhex[lenbuffer[0]] << 4;
136 r |= fromhex[lenbuffer[1]];
137 r <<= 4;
139 r |= fromhex[lenbuffer[2]];
140 r <<= 4;
142 r |= fromhex[lenbuffer[3]];
143 if (r < 0)
144 throw new ArrayIndexOutOfBoundsException();
145 return r;
146 } catch (ArrayIndexOutOfBoundsException err) {
147 throw new IOException("Invalid packet line header: "
148 + (char) lenbuffer[0] + (char) lenbuffer[1]
149 + (char) lenbuffer[2] + (char) lenbuffer[3]);