HADOOP-8507. Avoid OOM while deserializing DelegationTokenIdentifer. Contributed by Colin Patrick McCabe

git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/trunk@1349561 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Eli Collins 2012-06-12 22:10:12 +00:00
parent c83ccc5002
commit 0199fe9763
7 changed files with 115 additions and 29 deletions

View File

@ -245,6 +245,9 @@ Branch-2 ( Unreleased changes )
HADOOP-8485. Don't hardcode "Apache Hadoop 0.23" in the docs. (eli) HADOOP-8485. Don't hardcode "Apache Hadoop 0.23" in the docs. (eli)
HADOOP-8507. Avoid OOM while deserializing DelegationTokenIdentifer.
(Colin Patrick McCabe via eli)
BREAKDOWN OF HDFS-3042 SUBTASKS BREAKDOWN OF HDFS-3042 SUBTASKS
HADOOP-8220. ZKFailoverController doesn't handle failure to become active HADOOP-8220. ZKFailoverController doesn't handle failure to become active

View File

@ -254,7 +254,7 @@ public class FileStatus implements Writable, Comparable {
// Writable // Writable
////////////////////////////////////////////////// //////////////////////////////////////////////////
public void write(DataOutput out) throws IOException { public void write(DataOutput out) throws IOException {
Text.writeString(out, getPath().toString(), Text.ONE_MEGABYTE); Text.writeString(out, getPath().toString(), Text.DEFAULT_MAX_LEN);
out.writeLong(getLen()); out.writeLong(getLen());
out.writeBoolean(isDirectory()); out.writeBoolean(isDirectory());
out.writeShort(getReplication()); out.writeShort(getReplication());
@ -262,16 +262,16 @@ public class FileStatus implements Writable, Comparable {
out.writeLong(getModificationTime()); out.writeLong(getModificationTime());
out.writeLong(getAccessTime()); out.writeLong(getAccessTime());
getPermission().write(out); getPermission().write(out);
Text.writeString(out, getOwner(), Text.ONE_MEGABYTE); Text.writeString(out, getOwner(), Text.DEFAULT_MAX_LEN);
Text.writeString(out, getGroup(), Text.ONE_MEGABYTE); Text.writeString(out, getGroup(), Text.DEFAULT_MAX_LEN);
out.writeBoolean(isSymlink()); out.writeBoolean(isSymlink());
if (isSymlink()) { if (isSymlink()) {
Text.writeString(out, getSymlink().toString(), Text.ONE_MEGABYTE); Text.writeString(out, getSymlink().toString(), Text.DEFAULT_MAX_LEN);
} }
} }
public void readFields(DataInput in) throws IOException { public void readFields(DataInput in) throws IOException {
String strPath = Text.readString(in, Text.ONE_MEGABYTE); String strPath = Text.readString(in, Text.DEFAULT_MAX_LEN);
this.path = new Path(strPath); this.path = new Path(strPath);
this.length = in.readLong(); this.length = in.readLong();
this.isdir = in.readBoolean(); this.isdir = in.readBoolean();
@ -280,10 +280,10 @@ public class FileStatus implements Writable, Comparable {
modification_time = in.readLong(); modification_time = in.readLong();
access_time = in.readLong(); access_time = in.readLong();
permission.readFields(in); permission.readFields(in);
owner = Text.readString(in, Text.ONE_MEGABYTE); owner = Text.readString(in, Text.DEFAULT_MAX_LEN);
group = Text.readString(in, Text.ONE_MEGABYTE); group = Text.readString(in, Text.DEFAULT_MAX_LEN);
if (in.readBoolean()) { if (in.readBoolean()) {
this.symlink = new Path(Text.readString(in, Text.ONE_MEGABYTE)); this.symlink = new Path(Text.readString(in, Text.DEFAULT_MAX_LEN));
} else { } else {
this.symlink = null; this.symlink = null;
} }

View File

@ -84,8 +84,8 @@ public class PermissionStatus implements Writable {
/** {@inheritDoc} */ /** {@inheritDoc} */
public void readFields(DataInput in) throws IOException { public void readFields(DataInput in) throws IOException {
username = Text.readString(in, Text.ONE_MEGABYTE); username = Text.readString(in, Text.DEFAULT_MAX_LEN);
groupname = Text.readString(in, Text.ONE_MEGABYTE); groupname = Text.readString(in, Text.DEFAULT_MAX_LEN);
permission = FsPermission.read(in); permission = FsPermission.read(in);
} }
@ -110,8 +110,8 @@ public class PermissionStatus implements Writable {
String username, String username,
String groupname, String groupname,
FsPermission permission) throws IOException { FsPermission permission) throws IOException {
Text.writeString(out, username, Text.ONE_MEGABYTE); Text.writeString(out, username, Text.DEFAULT_MAX_LEN);
Text.writeString(out, groupname, Text.ONE_MEGABYTE); Text.writeString(out, groupname, Text.DEFAULT_MAX_LEN);
permission.write(out); permission.write(out);
} }

View File

@ -287,6 +287,20 @@ public class Text extends BinaryComparable
in.readFully(bytes, 0, newLength); in.readFully(bytes, 0, newLength);
length = newLength; length = newLength;
} }
public void readFields(DataInput in, int maxLength) throws IOException {
int newLength = WritableUtils.readVInt(in);
if (newLength < 0) {
throw new IOException("tried to deserialize " + newLength +
" bytes of data! newLength must be non-negative.");
} else if (newLength >= maxLength) {
throw new IOException("tried to deserialize " + newLength +
" bytes of data, but maxLength = " + maxLength);
}
setCapacity(newLength, false);
in.readFully(bytes, 0, newLength);
length = newLength;
}
/** Skips over one Text in the input. */ /** Skips over one Text in the input. */
public static void skip(DataInput in) throws IOException { public static void skip(DataInput in) throws IOException {
@ -304,6 +318,16 @@ public class Text extends BinaryComparable
out.write(bytes, 0, length); out.write(bytes, 0, length);
} }
public void write(DataOutput out, int maxLength) throws IOException {
if (length > maxLength) {
throw new IOException("data was too long to write! Expected " +
"less than or equal to " + maxLength + " bytes, but got " +
length + " bytes.");
}
WritableUtils.writeVInt(out, length);
out.write(bytes, 0, length);
}
/** Returns true iff <code>o</code> is a Text with the same contents. */ /** Returns true iff <code>o</code> is a Text with the same contents. */
public boolean equals(Object o) { public boolean equals(Object o) {
if (o instanceof Text) if (o instanceof Text)
@ -417,7 +441,7 @@ public class Text extends BinaryComparable
return bytes; return bytes;
} }
static final public int ONE_MEGABYTE = 1024 * 1024; static final public int DEFAULT_MAX_LEN = 1024 * 1024;
/** Read a UTF8 encoded string from in /** Read a UTF8 encoded string from in
*/ */
@ -432,7 +456,7 @@ public class Text extends BinaryComparable
*/ */
public static String readString(DataInput in, int maxLength) public static String readString(DataInput in, int maxLength)
throws IOException { throws IOException {
int length = WritableUtils.readVIntInRange(in, 0, maxLength - 1); int length = WritableUtils.readVIntInRange(in, 0, maxLength);
byte [] bytes = new byte[length]; byte [] bytes = new byte[length];
in.readFully(bytes, 0, length); in.readFully(bytes, 0, length);
return decode(bytes); return decode(bytes);
@ -454,9 +478,9 @@ public class Text extends BinaryComparable
throws IOException { throws IOException {
ByteBuffer bytes = encode(s); ByteBuffer bytes = encode(s);
int length = bytes.limit(); int length = bytes.limit();
if (length >= maxLength) { if (length > maxLength) {
throw new IOException("string was too long to write! Expected " + throw new IOException("string was too long to write! Expected " +
"less than " + maxLength + " bytes, but got " + "less than or equal to " + maxLength + " bytes, but got " +
length + " bytes."); length + " bytes.");
} }
WritableUtils.writeVInt(out, length); WritableUtils.writeVInt(out, length);

View File

@ -31,6 +31,8 @@ import org.apache.hadoop.security.HadoopKerberosName;
import org.apache.hadoop.security.UserGroupInformation; import org.apache.hadoop.security.UserGroupInformation;
import org.apache.hadoop.security.token.TokenIdentifier; import org.apache.hadoop.security.token.TokenIdentifier;
import com.google.common.annotations.VisibleForTesting;
@InterfaceAudience.LimitedPrivate({"HDFS", "MapReduce"}) @InterfaceAudience.LimitedPrivate({"HDFS", "MapReduce"})
@InterfaceStability.Evolving @InterfaceStability.Evolving
public abstract class AbstractDelegationTokenIdentifier public abstract class AbstractDelegationTokenIdentifier
@ -173,16 +175,17 @@ extends TokenIdentifier {
throw new IOException("Unknown version of delegation token " + throw new IOException("Unknown version of delegation token " +
version); version);
} }
owner.readFields(in); owner.readFields(in, Text.DEFAULT_MAX_LEN);
renewer.readFields(in); renewer.readFields(in, Text.DEFAULT_MAX_LEN);
realUser.readFields(in); realUser.readFields(in, Text.DEFAULT_MAX_LEN);
issueDate = WritableUtils.readVLong(in); issueDate = WritableUtils.readVLong(in);
maxDate = WritableUtils.readVLong(in); maxDate = WritableUtils.readVLong(in);
sequenceNumber = WritableUtils.readVInt(in); sequenceNumber = WritableUtils.readVInt(in);
masterKeyId = WritableUtils.readVInt(in); masterKeyId = WritableUtils.readVInt(in);
} }
public void write(DataOutput out) throws IOException { @VisibleForTesting
void writeImpl(DataOutput out) throws IOException {
out.writeByte(VERSION); out.writeByte(VERSION);
owner.write(out); owner.write(out);
renewer.write(out); renewer.write(out);
@ -193,6 +196,19 @@ extends TokenIdentifier {
WritableUtils.writeVInt(out, masterKeyId); WritableUtils.writeVInt(out, masterKeyId);
} }
public void write(DataOutput out) throws IOException {
if (owner.getLength() > Text.DEFAULT_MAX_LEN) {
throw new IOException("owner is too long to be serialized!");
}
if (renewer.getLength() > Text.DEFAULT_MAX_LEN) {
throw new IOException("renewer is too long to be serialized!");
}
if (realUser.getLength() > Text.DEFAULT_MAX_LEN) {
throw new IOException("realuser is too long to be serialized!");
}
writeImpl(out);
}
public String toString() { public String toString() {
StringBuilder buffer = new StringBuilder(); StringBuilder buffer = new StringBuilder();
buffer buffer

View File

@ -137,38 +137,38 @@ public class TestText extends TestCase {
} }
} }
public void doTestLimitedIO(String str, int strLen) throws IOException { public void doTestLimitedIO(String str, int len) throws IOException {
DataOutputBuffer out = new DataOutputBuffer(); DataOutputBuffer out = new DataOutputBuffer();
DataInputBuffer in = new DataInputBuffer(); DataInputBuffer in = new DataInputBuffer();
out.reset(); out.reset();
try { try {
Text.writeString(out, str, strLen); Text.writeString(out, str, len);
fail("expected writeString to fail when told to write a string " + fail("expected writeString to fail when told to write a string " +
"that was too long! The string was '" + str + "'"); "that was too long! The string was '" + str + "'");
} catch (IOException e) { } catch (IOException e) {
} }
Text.writeString(out, str, strLen + 1); Text.writeString(out, str, len + 1);
// test that it reads correctly // test that it reads correctly
in.reset(out.getData(), out.getLength()); in.reset(out.getData(), out.getLength());
in.mark(strLen); in.mark(len);
String after; String after;
try { try {
after = Text.readString(in, strLen); after = Text.readString(in, len);
fail("expected readString to fail when told to read a string " + fail("expected readString to fail when told to read a string " +
"that was too long! The string was '" + str + "'"); "that was too long! The string was '" + str + "'");
} catch (IOException e) { } catch (IOException e) {
} }
in.reset(); in.reset();
after = Text.readString(in, strLen + 1); after = Text.readString(in, len + 1);
assertTrue(str.equals(after)); assertTrue(str.equals(after));
} }
public void testLimitedIO() throws Exception { public void testLimitedIO() throws Exception {
doTestLimitedIO("abcd", 4); doTestLimitedIO("abcd", 3);
doTestLimitedIO("", 0); doTestLimitedIO("foo bar baz", 10);
doTestLimitedIO("1", 1); doTestLimitedIO("1", 0);
} }
public void testCompare() throws Exception { public void testCompare() throws Exception {

View File

@ -19,6 +19,7 @@
package org.apache.hadoop.security.token.delegation; package org.apache.hadoop.security.token.delegation;
import java.io.ByteArrayInputStream; import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.DataInput; import java.io.DataInput;
import java.io.DataInputStream; import java.io.DataInputStream;
import java.io.DataOutput; import java.io.DataOutput;
@ -387,4 +388,46 @@ public class TestDelegationToken {
} }
} }
private boolean testDelegationTokenIdentiferSerializationRoundTrip(Text owner,
Text renewer, Text realUser) throws IOException {
TestDelegationTokenIdentifier dtid = new TestDelegationTokenIdentifier(
owner, renewer, realUser);
DataOutputBuffer out = new DataOutputBuffer();
dtid.writeImpl(out);
DataInputBuffer in = new DataInputBuffer();
in.reset(out.getData(), out.getLength());
try {
TestDelegationTokenIdentifier dtid2 =
new TestDelegationTokenIdentifier();
dtid2.readFields(in);
assertTrue(dtid.equals(dtid2));
return true;
} catch(IOException e){
return false;
}
}
@Test
public void testSimpleDtidSerialization() throws IOException {
assertTrue(testDelegationTokenIdentiferSerializationRoundTrip(
new Text("owner"), new Text("renewer"), new Text("realUser")));
assertTrue(testDelegationTokenIdentiferSerializationRoundTrip(
new Text(""), new Text(""), new Text("")));
assertTrue(testDelegationTokenIdentiferSerializationRoundTrip(
new Text(""), new Text("b"), new Text("")));
}
@Test
public void testOverlongDtidSerialization() throws IOException {
byte[] bigBuf = new byte[Text.DEFAULT_MAX_LEN + 1];
for (int i = 0; i < bigBuf.length; i++) {
bigBuf[i] = 0;
}
assertFalse(testDelegationTokenIdentiferSerializationRoundTrip(
new Text(bigBuf), new Text("renewer"), new Text("realUser")));
assertFalse(testDelegationTokenIdentiferSerializationRoundTrip(
new Text("owner"), new Text(bigBuf), new Text("realUser")));
assertFalse(testDelegationTokenIdentiferSerializationRoundTrip(
new Text("owner"), new Text("renewer"), new Text(bigBuf)));
}
} }