HDFS-5737. Replacing only the default ACL can fail to copy unspecified base entries from the access ACL. Contributed by Chris Nauroth.

git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/branches/HDFS-4685@1556652 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Chris Nauroth 2014-01-08 22:02:53 +00:00
parent 2fbb3d694e
commit 023c11ec7e
3 changed files with 67 additions and 23 deletions

View File

@ -31,3 +31,6 @@ HDFS-4685 (Unreleased)
OPTIMIZATIONS OPTIMIZATIONS
BUG FIXES BUG FIXES
HDFS-5737. Replacing only the default ACL can fail to copy unspecified base
entries from the access ACL. (cnauroth)

View File

@ -62,6 +62,7 @@
@InterfaceAudience.Private @InterfaceAudience.Private
final class AclTransformation { final class AclTransformation {
private static final int MAX_ENTRIES = 32; private static final int MAX_ENTRIES = 32;
private static final int PIVOT_NOT_FOUND = -1;
/** /**
* Filters (discards) any existing ACL entries that have the same scope, type * Filters (discards) any existing ACL entries that have the same scope, type
@ -276,7 +277,7 @@ private static List<AclEntry> buildAndValidateAcl(
} }
aclBuilder.trimToSize(); aclBuilder.trimToSize();
Collections.sort(aclBuilder, ACL_ENTRY_COMPARATOR); Collections.sort(aclBuilder, ACL_ENTRY_COMPARATOR);
AclEntry userEntry = null, groupEntry = null, otherEntry = null; // Full iteration to check for duplicates and invalid named entries.
AclEntry prevEntry = null; AclEntry prevEntry = null;
for (AclEntry entry: aclBuilder) { for (AclEntry entry: aclBuilder) {
if (prevEntry != null && if (prevEntry != null &&
@ -289,22 +290,28 @@ private static List<AclEntry> buildAndValidateAcl(
throw new AclException( throw new AclException(
"Invalid ACL: this entry type must not have a name: " + entry + "."); "Invalid ACL: this entry type must not have a name: " + entry + ".");
} }
if (entry.getScope() == ACCESS) {
if (entry.getType() == USER && entry.getName() == null) {
userEntry = entry;
}
if (entry.getType() == GROUP && entry.getName() == null) {
groupEntry = entry;
}
if (entry.getType() == OTHER && entry.getName() == null) {
otherEntry = entry;
}
}
prevEntry = entry; prevEntry = entry;
} }
if (userEntry == null || groupEntry == null || otherEntry == null) { // Search for the required base access entries. If there is a default ACL,
throw new AclException( // then do the same check on the default entries.
"Invalid ACL: the user, group and other entries are required."); int pivot = calculatePivotOnDefaultEntries(aclBuilder);
for (AclEntryType type: EnumSet.of(USER, GROUP, OTHER)) {
AclEntry accessEntryKey = new AclEntry.Builder().setScope(ACCESS)
.setType(type).build();
if (Collections.binarySearch(aclBuilder, accessEntryKey,
ACL_ENTRY_COMPARATOR) < 0) {
throw new AclException(
"Invalid ACL: the user, group and other entries are required.");
}
if (pivot != PIVOT_NOT_FOUND) {
AclEntry defaultEntryKey = new AclEntry.Builder().setScope(DEFAULT)
.setType(type).build();
if (Collections.binarySearch(aclBuilder, defaultEntryKey,
ACL_ENTRY_COMPARATOR) < 0) {
throw new AclException(
"Invalid default ACL: the user, group and other entries are required.");
}
}
} }
return Collections.unmodifiableList(aclBuilder); return Collections.unmodifiableList(aclBuilder);
} }
@ -372,6 +379,23 @@ private static void calculateMasks(List<AclEntry> aclBuilder,
} }
} }
/**
* Returns the pivot point in the list between the access entries and the
* default entries. This is the index of the first element in the list that is
* a default entry.
*
* @param aclBuilder ArrayList<AclEntry> containing entries to build
* @return int pivot point, or -1 if list contains no default entries
*/
private static int calculatePivotOnDefaultEntries(List<AclEntry> aclBuilder) {
for (int i = 0; i < aclBuilder.size(); ++i) {
if (aclBuilder.get(i).getScope() == DEFAULT) {
return i;
}
}
return PIVOT_NOT_FOUND;
}
/** /**
* Adds unspecified default entries by copying permissions from the * Adds unspecified default entries by copying permissions from the
* corresponding access entries. * corresponding access entries.
@ -379,14 +403,9 @@ private static void calculateMasks(List<AclEntry> aclBuilder,
* @param aclBuilder ArrayList<AclEntry> containing entries to build * @param aclBuilder ArrayList<AclEntry> containing entries to build
*/ */
private static void copyDefaultsIfNeeded(List<AclEntry> aclBuilder) { private static void copyDefaultsIfNeeded(List<AclEntry> aclBuilder) {
int pivot = -1; Collections.sort(aclBuilder, ACL_ENTRY_COMPARATOR);
for (int i = 0; i < aclBuilder.size(); ++i) { int pivot = calculatePivotOnDefaultEntries(aclBuilder);
if (aclBuilder.get(i).getScope() == DEFAULT) { if (pivot != PIVOT_NOT_FOUND) {
pivot = i;
break;
}
}
if (pivot > -1) {
List<AclEntry> accessEntries = aclBuilder.subList(0, pivot); List<AclEntry> accessEntries = aclBuilder.subList(0, pivot);
List<AclEntry> defaultEntries = aclBuilder.subList(pivot, List<AclEntry> defaultEntries = aclBuilder.subList(pivot,
aclBuilder.size()); aclBuilder.size());

View File

@ -1058,6 +1058,28 @@ public void testReplaceAclEntriesAutomaticDefaultOther() throws AclException {
assertEquals(expected, replaceAclEntries(existing, aclSpec)); assertEquals(expected, replaceAclEntries(existing, aclSpec));
} }
@Test
public void testReplaceAclEntriesOnlyDefaults() throws AclException {
List<AclEntry> existing = new ImmutableList.Builder<AclEntry>()
.add(aclEntry(ACCESS, USER, ALL))
.add(aclEntry(ACCESS, GROUP, READ))
.add(aclEntry(ACCESS, OTHER, NONE))
.build();
List<AclEntry> aclSpec = Lists.newArrayList(
aclEntry(DEFAULT, USER, "bruce", READ));
List<AclEntry> expected = new ImmutableList.Builder<AclEntry>()
.add(aclEntry(ACCESS, USER, ALL))
.add(aclEntry(ACCESS, GROUP, READ))
.add(aclEntry(ACCESS, OTHER, NONE))
.add(aclEntry(DEFAULT, USER, ALL))
.add(aclEntry(DEFAULT, USER, "bruce", READ))
.add(aclEntry(DEFAULT, GROUP, READ))
.add(aclEntry(DEFAULT, MASK, READ))
.add(aclEntry(DEFAULT, OTHER, NONE))
.build();
assertEquals(expected, replaceAclEntries(existing, aclSpec));
}
@Test(expected=AclException.class) @Test(expected=AclException.class)
public void testReplaceAclEntriesInputTooLarge() throws AclException { public void testReplaceAclEntriesInputTooLarge() throws AclException {
List<AclEntry> existing = new ImmutableList.Builder<AclEntry>() List<AclEntry> existing = new ImmutableList.Builder<AclEntry>()