HDFS-6422. getfattr in CLI doesn't throw exception or return non-0 return code when xattr doesn't exist. (Charles Lamb via umamahesh)
git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/trunk@1612922 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
b9370fdcf6
commit
5343b43fd9
@ -892,6 +892,9 @@ Release 2.5.0 - UNRELEASED
|
|||||||
HDFS-6703. NFS: Files can be deleted from a read-only mount
|
HDFS-6703. NFS: Files can be deleted from a read-only mount
|
||||||
(Srikanth Upputuri via brandonli)
|
(Srikanth Upputuri via brandonli)
|
||||||
|
|
||||||
|
HDFS-6422. getfattr in CLI doesn't throw exception or return non-0 return code
|
||||||
|
when xattr doesn't exist. (Charles Lamb via umamahesh)
|
||||||
|
|
||||||
BREAKDOWN OF HDFS-2006 SUBTASKS AND RELATED JIRAS
|
BREAKDOWN OF HDFS-2006 SUBTASKS AND RELATED JIRAS
|
||||||
|
|
||||||
HDFS-6299. Protobuf for XAttr and client-side implementation. (Yi Liu via umamahesh)
|
HDFS-6299. Protobuf for XAttr and client-side implementation. (Yi Liu via umamahesh)
|
||||||
|
@ -1337,6 +1337,6 @@ public interface ClientProtocol {
|
|||||||
* @param xAttr <code>XAttr</code> to remove
|
* @param xAttr <code>XAttr</code> to remove
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
@Idempotent
|
@AtMostOnce
|
||||||
public void removeXAttr(String src, XAttr xAttr) throws IOException;
|
public void removeXAttr(String src, XAttr xAttr) throws IOException;
|
||||||
}
|
}
|
||||||
|
@ -1074,10 +1074,11 @@ public class FSEditLog implements LogsPurgeable {
|
|||||||
logEdit(op);
|
logEdit(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
void logRemoveXAttrs(String src, List<XAttr> xAttrs) {
|
void logRemoveXAttrs(String src, List<XAttr> xAttrs, boolean toLogRpcIds) {
|
||||||
final RemoveXAttrOp op = RemoveXAttrOp.getInstance();
|
final RemoveXAttrOp op = RemoveXAttrOp.getInstance();
|
||||||
op.src = src;
|
op.src = src;
|
||||||
op.xAttrs = xAttrs;
|
op.xAttrs = xAttrs;
|
||||||
|
logRpcIds(op, toLogRpcIds);
|
||||||
logEdit(op);
|
logEdit(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -821,6 +821,10 @@ public class FSEditLogLoader {
|
|||||||
RemoveXAttrOp removeXAttrOp = (RemoveXAttrOp) op;
|
RemoveXAttrOp removeXAttrOp = (RemoveXAttrOp) op;
|
||||||
fsDir.unprotectedRemoveXAttrs(removeXAttrOp.src,
|
fsDir.unprotectedRemoveXAttrs(removeXAttrOp.src,
|
||||||
removeXAttrOp.xAttrs);
|
removeXAttrOp.xAttrs);
|
||||||
|
if (toAddRetryCache) {
|
||||||
|
fsNamesys.addCacheEntry(removeXAttrOp.rpcClientId,
|
||||||
|
removeXAttrOp.rpcCallId);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
|
@ -3551,6 +3551,7 @@ public abstract class FSEditLogOp {
|
|||||||
XAttrEditLogProto p = XAttrEditLogProto.parseDelimitedFrom(in);
|
XAttrEditLogProto p = XAttrEditLogProto.parseDelimitedFrom(in);
|
||||||
src = p.getSrc();
|
src = p.getSrc();
|
||||||
xAttrs = PBHelper.convertXAttrs(p.getXAttrsList());
|
xAttrs = PBHelper.convertXAttrs(p.getXAttrsList());
|
||||||
|
readRpcIds(in, logVersion);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -3561,18 +3562,22 @@ public abstract class FSEditLogOp {
|
|||||||
}
|
}
|
||||||
b.addAllXAttrs(PBHelper.convertXAttrProto(xAttrs));
|
b.addAllXAttrs(PBHelper.convertXAttrProto(xAttrs));
|
||||||
b.build().writeDelimitedTo(out);
|
b.build().writeDelimitedTo(out);
|
||||||
|
// clientId and callId
|
||||||
|
writeRpcIds(rpcClientId, rpcCallId, out);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void toXml(ContentHandler contentHandler) throws SAXException {
|
protected void toXml(ContentHandler contentHandler) throws SAXException {
|
||||||
XMLUtils.addSaxString(contentHandler, "SRC", src);
|
XMLUtils.addSaxString(contentHandler, "SRC", src);
|
||||||
appendXAttrsToXml(contentHandler, xAttrs);
|
appendXAttrsToXml(contentHandler, xAttrs);
|
||||||
|
appendRpcIdsToXml(contentHandler, rpcClientId, rpcCallId);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
void fromXml(Stanza st) throws InvalidXmlException {
|
void fromXml(Stanza st) throws InvalidXmlException {
|
||||||
src = st.getValue("SRC");
|
src = st.getValue("SRC");
|
||||||
xAttrs = readXAttrsFromXml(st);
|
xAttrs = readXAttrsFromXml(st);
|
||||||
|
readRpcIdsFromXml(st);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8279,11 +8279,12 @@ public class FSNamesystem implements Namesystem, FSClusterStats,
|
|||||||
nnConf.checkXAttrsConfigFlag();
|
nnConf.checkXAttrsConfigFlag();
|
||||||
FSPermissionChecker pc = getPermissionChecker();
|
FSPermissionChecker pc = getPermissionChecker();
|
||||||
boolean getAll = xAttrs == null || xAttrs.isEmpty();
|
boolean getAll = xAttrs == null || xAttrs.isEmpty();
|
||||||
List<XAttr> filteredXAttrs = null;
|
|
||||||
if (!getAll) {
|
if (!getAll) {
|
||||||
filteredXAttrs = XAttrPermissionFilter.filterXAttrsForApi(pc, xAttrs);
|
try {
|
||||||
if (filteredXAttrs.isEmpty()) {
|
XAttrPermissionFilter.checkPermissionForApi(pc, xAttrs);
|
||||||
return filteredXAttrs;
|
} catch (AccessControlException e) {
|
||||||
|
logAuditEvent(false, "getXAttrs", src);
|
||||||
|
throw e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
checkOperation(OperationCategory.READ);
|
checkOperation(OperationCategory.READ);
|
||||||
@ -8302,15 +8303,21 @@ public class FSNamesystem implements Namesystem, FSClusterStats,
|
|||||||
if (filteredAll == null || filteredAll.isEmpty()) {
|
if (filteredAll == null || filteredAll.isEmpty()) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
List<XAttr> toGet = Lists.newArrayListWithCapacity(filteredXAttrs.size());
|
List<XAttr> toGet = Lists.newArrayListWithCapacity(xAttrs.size());
|
||||||
for (XAttr xAttr : filteredXAttrs) {
|
for (XAttr xAttr : xAttrs) {
|
||||||
|
boolean foundIt = false;
|
||||||
for (XAttr a : filteredAll) {
|
for (XAttr a : filteredAll) {
|
||||||
if (xAttr.getNameSpace() == a.getNameSpace()
|
if (xAttr.getNameSpace() == a.getNameSpace()
|
||||||
&& xAttr.getName().equals(a.getName())) {
|
&& xAttr.getName().equals(a.getName())) {
|
||||||
toGet.add(a);
|
toGet.add(a);
|
||||||
|
foundIt = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (!foundIt) {
|
||||||
|
throw new IOException(
|
||||||
|
"At least one of the attributes provided was not found.");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return toGet;
|
return toGet;
|
||||||
}
|
}
|
||||||
@ -8345,16 +8352,41 @@ public class FSNamesystem implements Namesystem, FSClusterStats,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove an xattr for a file or directory.
|
||||||
|
*
|
||||||
|
* @param src
|
||||||
|
* - path to remove the xattr from
|
||||||
|
* @param xAttr
|
||||||
|
* - xAttr to remove
|
||||||
|
* @throws AccessControlException
|
||||||
|
* @throws SafeModeException
|
||||||
|
* @throws UnresolvedLinkException
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
void removeXAttr(String src, XAttr xAttr) throws IOException {
|
void removeXAttr(String src, XAttr xAttr) throws IOException {
|
||||||
nnConf.checkXAttrsConfigFlag();
|
CacheEntry cacheEntry = RetryCache.waitForCompletion(retryCache);
|
||||||
HdfsFileStatus resultingStat = null;
|
if (cacheEntry != null && cacheEntry.isSuccess()) {
|
||||||
FSPermissionChecker pc = getPermissionChecker();
|
return; // Return previous response
|
||||||
|
}
|
||||||
|
boolean success = false;
|
||||||
try {
|
try {
|
||||||
XAttrPermissionFilter.checkPermissionForApi(pc, xAttr);
|
removeXAttrInt(src, xAttr, cacheEntry != null);
|
||||||
|
success = true;
|
||||||
} catch (AccessControlException e) {
|
} catch (AccessControlException e) {
|
||||||
logAuditEvent(false, "removeXAttr", src);
|
logAuditEvent(false, "removeXAttr", src);
|
||||||
throw e;
|
throw e;
|
||||||
|
} finally {
|
||||||
|
RetryCache.setState(cacheEntry, success);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void removeXAttrInt(String src, XAttr xAttr, boolean logRetryCache)
|
||||||
|
throws IOException {
|
||||||
|
nnConf.checkXAttrsConfigFlag();
|
||||||
|
HdfsFileStatus resultingStat = null;
|
||||||
|
FSPermissionChecker pc = getPermissionChecker();
|
||||||
|
XAttrPermissionFilter.checkPermissionForApi(pc, xAttr);
|
||||||
checkOperation(OperationCategory.WRITE);
|
checkOperation(OperationCategory.WRITE);
|
||||||
byte[][] pathComponents = FSDirectory.getPathComponentsForReservedPath(src);
|
byte[][] pathComponents = FSDirectory.getPathComponentsForReservedPath(src);
|
||||||
writeLock();
|
writeLock();
|
||||||
@ -8368,12 +8400,12 @@ public class FSNamesystem implements Namesystem, FSClusterStats,
|
|||||||
xAttrs.add(xAttr);
|
xAttrs.add(xAttr);
|
||||||
List<XAttr> removedXAttrs = dir.removeXAttrs(src, xAttrs);
|
List<XAttr> removedXAttrs = dir.removeXAttrs(src, xAttrs);
|
||||||
if (removedXAttrs != null && !removedXAttrs.isEmpty()) {
|
if (removedXAttrs != null && !removedXAttrs.isEmpty()) {
|
||||||
getEditLog().logRemoveXAttrs(src, removedXAttrs);
|
getEditLog().logRemoveXAttrs(src, removedXAttrs, logRetryCache);
|
||||||
|
} else {
|
||||||
|
throw new IOException(
|
||||||
|
"No matching attributes found for remove operation");
|
||||||
}
|
}
|
||||||
resultingStat = getAuditFileInfo(src, false);
|
resultingStat = getAuditFileInfo(src, false);
|
||||||
} catch (AccessControlException e) {
|
|
||||||
logAuditEvent(false, "removeXAttr", src);
|
|
||||||
throw e;
|
|
||||||
} finally {
|
} finally {
|
||||||
writeUnlock();
|
writeUnlock();
|
||||||
}
|
}
|
||||||
|
@ -26,6 +26,7 @@ import org.apache.hadoop.hdfs.XAttrHelper;
|
|||||||
import org.apache.hadoop.security.AccessControlException;
|
import org.apache.hadoop.security.AccessControlException;
|
||||||
|
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* There are four types of extended attributes <XAttr> defined by the
|
* There are four types of extended attributes <XAttr> defined by the
|
||||||
@ -61,6 +62,18 @@ public class XAttrPermissionFilter {
|
|||||||
+ XAttrHelper.getPrefixName(xAttr));
|
+ XAttrHelper.getPrefixName(xAttr));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void checkPermissionForApi(FSPermissionChecker pc,
|
||||||
|
List<XAttr> xAttrs) throws AccessControlException {
|
||||||
|
Preconditions.checkArgument(xAttrs != null);
|
||||||
|
if (xAttrs.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (XAttr xAttr : xAttrs) {
|
||||||
|
checkPermissionForApi(pc, xAttr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static List<XAttr> filterXAttrsForApi(FSPermissionChecker pc,
|
static List<XAttr> filterXAttrsForApi(FSPermissionChecker pc,
|
||||||
List<XAttr> xAttrs) {
|
List<XAttr> xAttrs) {
|
||||||
assert xAttrs != null : "xAttrs can not be null";
|
assert xAttrs != null : "xAttrs can not be null";
|
||||||
|
@ -26,7 +26,7 @@ public class XAttrNameParam extends StringParam {
|
|||||||
public static final String DEFAULT = "";
|
public static final String DEFAULT = "";
|
||||||
|
|
||||||
private static Domain DOMAIN = new Domain(NAME,
|
private static Domain DOMAIN = new Domain(NAME,
|
||||||
Pattern.compile("^(user\\.|trusted\\.|system\\.|security\\.).+"));
|
Pattern.compile(".*"));
|
||||||
|
|
||||||
public XAttrNameParam(final String str) {
|
public XAttrNameParam(final String str) {
|
||||||
super(DOMAIN, str == null || str.equals(DEFAULT) ? null : str);
|
super(DOMAIN, str == null || str.equals(DEFAULT) ? null : str);
|
||||||
|
@ -2653,6 +2653,75 @@ public class TestDFSShell {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 1. Test that CLI throws an exception and returns non-0 when user does
|
||||||
|
* not have permission to read an xattr.
|
||||||
|
* 2. Test that CLI throws an exception and returns non-0 when a non-existent
|
||||||
|
* xattr is requested.
|
||||||
|
*/
|
||||||
|
@Test (timeout = 120000)
|
||||||
|
public void testGetFAttrErrors() throws Exception {
|
||||||
|
final UserGroupInformation user = UserGroupInformation.
|
||||||
|
createUserForTesting("user", new String[] {"mygroup"});
|
||||||
|
MiniDFSCluster cluster = null;
|
||||||
|
PrintStream bakErr = null;
|
||||||
|
try {
|
||||||
|
final Configuration conf = new HdfsConfiguration();
|
||||||
|
cluster = new MiniDFSCluster.Builder(conf).numDataNodes(1).build();
|
||||||
|
cluster.waitActive();
|
||||||
|
|
||||||
|
final FileSystem fs = cluster.getFileSystem();
|
||||||
|
final Path p = new Path("/foo");
|
||||||
|
fs.mkdirs(p);
|
||||||
|
bakErr = System.err;
|
||||||
|
|
||||||
|
final FsShell fshell = new FsShell(conf);
|
||||||
|
final ByteArrayOutputStream out = new ByteArrayOutputStream();
|
||||||
|
System.setErr(new PrintStream(out));
|
||||||
|
|
||||||
|
// No permission for "other".
|
||||||
|
fs.setPermission(p, new FsPermission((short) 0700));
|
||||||
|
|
||||||
|
{
|
||||||
|
final int ret = ToolRunner.run(fshell, new String[] {
|
||||||
|
"-setfattr", "-n", "user.a1", "-v", "1234", "/foo"});
|
||||||
|
assertEquals("Returned should be 0", 0, ret);
|
||||||
|
out.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
int ret = ToolRunner.run(fshell, new String[] {
|
||||||
|
"-getfattr", "-n", "user.a1", "/foo"});
|
||||||
|
String str = out.toString();
|
||||||
|
assertTrue("xattr value was incorrectly returned",
|
||||||
|
str.indexOf("1234") == -1);
|
||||||
|
out.reset();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
{
|
||||||
|
final int ret = ToolRunner.run(fshell, new String[]{
|
||||||
|
"-getfattr", "-n", "user.nonexistent", "/foo"});
|
||||||
|
String str = out.toString();
|
||||||
|
assertTrue("xattr value was incorrectly returned",
|
||||||
|
str.indexOf(
|
||||||
|
"getfattr: At least one of the attributes provided was not found")
|
||||||
|
>= 0);
|
||||||
|
out.reset();
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
if (bakErr != null) {
|
||||||
|
System.setErr(bakErr);
|
||||||
|
}
|
||||||
|
if (cluster != null) {
|
||||||
|
cluster.shutdown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test that the server trash configuration is respected when
|
* Test that the server trash configuration is respected when
|
||||||
* the client configuration is not set.
|
* the client configuration is not set.
|
||||||
|
@ -19,7 +19,6 @@ package org.apache.hadoop.hdfs.server.namenode;
|
|||||||
|
|
||||||
import java.io.FileNotFoundException;
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.FileNotFoundException;
|
|
||||||
import java.security.PrivilegedExceptionAction;
|
import java.security.PrivilegedExceptionAction;
|
||||||
import java.util.EnumSet;
|
import java.util.EnumSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@ -46,6 +45,7 @@ import static org.apache.hadoop.fs.permission.AclEntryType.USER;
|
|||||||
import static org.apache.hadoop.fs.permission.FsAction.ALL;
|
import static org.apache.hadoop.fs.permission.FsAction.ALL;
|
||||||
import static org.apache.hadoop.fs.permission.FsAction.READ;
|
import static org.apache.hadoop.fs.permission.FsAction.READ;
|
||||||
import static org.apache.hadoop.hdfs.server.namenode.AclTestHelpers.aclEntry;
|
import static org.apache.hadoop.hdfs.server.namenode.AclTestHelpers.aclEntry;
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
@ -261,11 +261,12 @@ public class FSXAttrBaseTest {
|
|||||||
fs.setXAttr(path, "user.", value1, EnumSet.of(XAttrSetFlag.CREATE,
|
fs.setXAttr(path, "user.", value1, EnumSet.of(XAttrSetFlag.CREATE,
|
||||||
XAttrSetFlag.REPLACE));
|
XAttrSetFlag.REPLACE));
|
||||||
Assert.fail("Setting xattr with empty name should fail.");
|
Assert.fail("Setting xattr with empty name should fail.");
|
||||||
|
} catch (RemoteException e) {
|
||||||
|
assertEquals("Unexpected RemoteException: " + e, e.getClassName(),
|
||||||
|
HadoopIllegalArgumentException.class.getCanonicalName());
|
||||||
|
GenericTestUtils.assertExceptionContains("XAttr name cannot be empty", e);
|
||||||
} catch (HadoopIllegalArgumentException e) {
|
} catch (HadoopIllegalArgumentException e) {
|
||||||
GenericTestUtils.assertExceptionContains("XAttr name cannot be empty", e);
|
GenericTestUtils.assertExceptionContains("XAttr name cannot be empty", e);
|
||||||
} catch (IllegalArgumentException e) {
|
|
||||||
GenericTestUtils.assertExceptionContains("Invalid value: \"user.\" does " +
|
|
||||||
"not belong to the domain ^(user\\.|trusted\\.|system\\.|security\\.).+", e);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set xattr with invalid name: "a1"
|
// Set xattr with invalid name: "a1"
|
||||||
@ -274,11 +275,12 @@ public class FSXAttrBaseTest {
|
|||||||
XAttrSetFlag.REPLACE));
|
XAttrSetFlag.REPLACE));
|
||||||
Assert.fail("Setting xattr with invalid name prefix or without " +
|
Assert.fail("Setting xattr with invalid name prefix or without " +
|
||||||
"name prefix should fail.");
|
"name prefix should fail.");
|
||||||
|
} catch (RemoteException e) {
|
||||||
|
assertEquals("Unexpected RemoteException: " + e, e.getClassName(),
|
||||||
|
HadoopIllegalArgumentException.class.getCanonicalName());
|
||||||
|
GenericTestUtils.assertExceptionContains("XAttr name must be prefixed", e);
|
||||||
} catch (HadoopIllegalArgumentException e) {
|
} catch (HadoopIllegalArgumentException e) {
|
||||||
GenericTestUtils.assertExceptionContains("XAttr name must be prefixed", e);
|
GenericTestUtils.assertExceptionContains("XAttr name must be prefixed", e);
|
||||||
} catch (IllegalArgumentException e) {
|
|
||||||
GenericTestUtils.assertExceptionContains("Invalid value: \"a1\" does " +
|
|
||||||
"not belong to the domain ^(user\\.|trusted\\.|system\\.|security\\.).+", e);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set xattr without XAttrSetFlag
|
// Set xattr without XAttrSetFlag
|
||||||
@ -341,9 +343,18 @@ public class FSXAttrBaseTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests for getting xattr
|
* getxattr tests. Test that getxattr throws an exception if any of
|
||||||
* 1. To get xattr which does not exist.
|
* the following are true:
|
||||||
* 2. To get multiple xattrs.
|
* an xattr that was requested doesn't exist
|
||||||
|
* the caller specifies an unknown namespace
|
||||||
|
* the caller doesn't have access to the namespace
|
||||||
|
* the caller doesn't have permission to get the value of the xattr
|
||||||
|
* the caller does not have search access to the parent directory
|
||||||
|
* the caller has only read access to the owning directory
|
||||||
|
* the caller has only search access to the owning directory and
|
||||||
|
* execute/search access to the actual entity
|
||||||
|
* the caller does not have search access to the owning directory and read
|
||||||
|
* access to the actual entity
|
||||||
*/
|
*/
|
||||||
@Test(timeout = 120000)
|
@Test(timeout = 120000)
|
||||||
public void testGetXAttrs() throws Exception {
|
public void testGetXAttrs() throws Exception {
|
||||||
@ -351,21 +362,159 @@ public class FSXAttrBaseTest {
|
|||||||
fs.setXAttr(path, name1, value1, EnumSet.of(XAttrSetFlag.CREATE));
|
fs.setXAttr(path, name1, value1, EnumSet.of(XAttrSetFlag.CREATE));
|
||||||
fs.setXAttr(path, name2, value2, EnumSet.of(XAttrSetFlag.CREATE));
|
fs.setXAttr(path, name2, value2, EnumSet.of(XAttrSetFlag.CREATE));
|
||||||
|
|
||||||
// XAttr does not exist.
|
/* An XAttr that was requested does not exist. */
|
||||||
byte[] value = fs.getXAttr(path, name3);
|
try {
|
||||||
Assert.assertEquals(value, null);
|
final byte[] value = fs.getXAttr(path, name3);
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains(
|
||||||
|
"At least one of the attributes provided was not found.", e);
|
||||||
|
}
|
||||||
|
|
||||||
List<String> names = Lists.newArrayList();
|
/* Throw an exception if an xattr that was requested does not exist. */
|
||||||
|
{
|
||||||
|
final List<String> names = Lists.newArrayList();
|
||||||
names.add(name1);
|
names.add(name1);
|
||||||
names.add(name2);
|
names.add(name2);
|
||||||
names.add(name3);
|
names.add(name3);
|
||||||
Map<String, byte[]> xattrs = fs.getXAttrs(path, names);
|
try {
|
||||||
Assert.assertEquals(xattrs.size(), 2);
|
final Map<String, byte[]> xattrs = fs.getXAttrs(path, names);
|
||||||
Assert.assertArrayEquals(value1, xattrs.get(name1));
|
Assert.fail("expected IOException");
|
||||||
Assert.assertArrayEquals(value2, xattrs.get(name2));
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains(
|
||||||
|
"At least one of the attributes provided was not found.", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fs.removeXAttr(path, name1);
|
fs.removeXAttr(path, name1);
|
||||||
fs.removeXAttr(path, name2);
|
fs.removeXAttr(path, name2);
|
||||||
|
|
||||||
|
/* Unknown namespace should throw an exception. */
|
||||||
|
try {
|
||||||
|
final byte[] xattr = fs.getXAttr(path, "wackynamespace.foo");
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (Exception e) {
|
||||||
|
GenericTestUtils.assertExceptionContains
|
||||||
|
("An XAttr name must be prefixed with user/trusted/security/system, " +
|
||||||
|
"followed by a '.'",
|
||||||
|
e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The 'trusted' namespace should not be accessible and should throw an
|
||||||
|
* exception.
|
||||||
|
*/
|
||||||
|
final UserGroupInformation user = UserGroupInformation.
|
||||||
|
createUserForTesting("user", new String[] {"mygroup"});
|
||||||
|
fs.setXAttr(path, "trusted.foo", "1234".getBytes());
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
final byte[] xattr = userFs.getXAttr(path, "trusted.foo");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("User doesn't have permission", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.setXAttr(path, name1, "1234".getBytes());
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Test that an exception is thrown if the caller doesn't have permission to
|
||||||
|
* get the value of the xattr.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Set access so that only the owner has access. */
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0700));
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
final byte[] xattr = userFs.getXAttr(path, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The caller must have search access to the parent directory.
|
||||||
|
*/
|
||||||
|
final Path childDir = new Path(path, "child" + pathCount);
|
||||||
|
/* Set access to parent so that only the owner has access. */
|
||||||
|
FileSystem.mkdirs(fs, childDir, FsPermission.createImmutable((short)0700));
|
||||||
|
fs.setXAttr(childDir, name1, "1234".getBytes());
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
final byte[] xattr = userFs.getXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check that read access to the owning directory is not good enough. */
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0704));
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
final byte[] xattr = userFs.getXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check that search access to the owning directory and search/execute
|
||||||
|
* access to the actual entity with extended attributes is not good enough.
|
||||||
|
*/
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0701));
|
||||||
|
fs.setPermission(childDir, new FsPermission((short) 0701));
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
final byte[] xattr = userFs.getXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check that search access to the owning directory and read access to
|
||||||
|
* the actual entity with the extended attribute is good enough.
|
||||||
|
*/
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0701));
|
||||||
|
fs.setPermission(childDir, new FsPermission((short) 0704));
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
final byte[] xattr = userFs.getXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -402,6 +551,166 @@ public class FSXAttrBaseTest {
|
|||||||
fs.removeXAttr(path, name3);
|
fs.removeXAttr(path, name3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* removexattr tests. Test that removexattr throws an exception if any of
|
||||||
|
* the following are true:
|
||||||
|
* an xattr that was requested doesn't exist
|
||||||
|
* the caller specifies an unknown namespace
|
||||||
|
* the caller doesn't have access to the namespace
|
||||||
|
* the caller doesn't have permission to get the value of the xattr
|
||||||
|
* the caller does not have "execute" (scan) access to the parent directory
|
||||||
|
* the caller has only read access to the owning directory
|
||||||
|
* the caller has only execute access to the owning directory and execute
|
||||||
|
* access to the actual entity
|
||||||
|
* the caller does not have execute access to the owning directory and write
|
||||||
|
* access to the actual entity
|
||||||
|
*/
|
||||||
|
@Test(timeout = 120000)
|
||||||
|
public void testRemoveXAttrPermissions() throws Exception {
|
||||||
|
FileSystem.mkdirs(fs, path, FsPermission.createImmutable((short)0750));
|
||||||
|
fs.setXAttr(path, name1, value1, EnumSet.of(XAttrSetFlag.CREATE));
|
||||||
|
fs.setXAttr(path, name2, value2, EnumSet.of(XAttrSetFlag.CREATE));
|
||||||
|
fs.setXAttr(path, name3, null, EnumSet.of(XAttrSetFlag.CREATE));
|
||||||
|
|
||||||
|
try {
|
||||||
|
fs.removeXAttr(path, name2);
|
||||||
|
fs.removeXAttr(path, name2);
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("No matching attributes found", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Unknown namespace should throw an exception. */
|
||||||
|
final String expectedExceptionString = "An XAttr name must be prefixed " +
|
||||||
|
"with user/trusted/security/system, followed by a '.'";
|
||||||
|
try {
|
||||||
|
fs.removeXAttr(path, "wackynamespace.foo");
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (RemoteException e) {
|
||||||
|
assertEquals("Unexpected RemoteException: " + e, e.getClassName(),
|
||||||
|
HadoopIllegalArgumentException.class.getCanonicalName());
|
||||||
|
GenericTestUtils.assertExceptionContains(expectedExceptionString, e);
|
||||||
|
} catch (HadoopIllegalArgumentException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains(expectedExceptionString, e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The 'trusted' namespace should not be accessible and should throw an
|
||||||
|
* exception.
|
||||||
|
*/
|
||||||
|
final UserGroupInformation user = UserGroupInformation.
|
||||||
|
createUserForTesting("user", new String[] {"mygroup"});
|
||||||
|
fs.setXAttr(path, "trusted.foo", "1234".getBytes());
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
userFs.removeXAttr(path, "trusted.foo");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("User doesn't have permission", e);
|
||||||
|
} finally {
|
||||||
|
fs.removeXAttr(path, "trusted.foo");
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Test that an exception is thrown if the caller doesn't have permission to
|
||||||
|
* get the value of the xattr.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Set access so that only the owner has access. */
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0700));
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
userFs.removeXAttr(path, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The caller must have "execute" (scan) access to the parent directory.
|
||||||
|
*/
|
||||||
|
final Path childDir = new Path(path, "child" + pathCount);
|
||||||
|
/* Set access to parent so that only the owner has access. */
|
||||||
|
FileSystem.mkdirs(fs, childDir, FsPermission.createImmutable((short)0700));
|
||||||
|
fs.setXAttr(childDir, name1, "1234".getBytes());
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
userFs.removeXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check that read access to the owning directory is not good enough. */
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0704));
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
userFs.removeXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check that execute access to the owning directory and scan access to
|
||||||
|
* the actual entity with extended attributes is not good enough.
|
||||||
|
*/
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0701));
|
||||||
|
fs.setPermission(childDir, new FsPermission((short) 0701));
|
||||||
|
try {
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
userFs.removeXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Assert.fail("expected IOException");
|
||||||
|
} catch (IOException e) {
|
||||||
|
GenericTestUtils.assertExceptionContains("Permission denied", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check that execute access to the owning directory and write access to
|
||||||
|
* the actual entity with extended attributes is good enough.
|
||||||
|
*/
|
||||||
|
fs.setPermission(path, new FsPermission((short) 0701));
|
||||||
|
fs.setPermission(childDir, new FsPermission((short) 0706));
|
||||||
|
user.doAs(new PrivilegedExceptionAction<Object>() {
|
||||||
|
@Override
|
||||||
|
public Object run() throws Exception {
|
||||||
|
final FileSystem userFs = dfsCluster.getFileSystem();
|
||||||
|
userFs.removeXAttr(childDir, name1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
@Test(timeout = 120000)
|
@Test(timeout = 120000)
|
||||||
public void testRenameFileWithXAttr() throws Exception {
|
public void testRenameFileWithXAttr() throws Exception {
|
||||||
FileSystem.mkdirs(fs, path, FsPermission.createImmutable((short)0750));
|
FileSystem.mkdirs(fs, path, FsPermission.createImmutable((short)0750));
|
||||||
|
@ -415,7 +415,7 @@ public class TestNamenodeRetryCache {
|
|||||||
|
|
||||||
LightWeightCache<CacheEntry, CacheEntry> cacheSet =
|
LightWeightCache<CacheEntry, CacheEntry> cacheSet =
|
||||||
(LightWeightCache<CacheEntry, CacheEntry>) namesystem.getRetryCache().getCacheSet();
|
(LightWeightCache<CacheEntry, CacheEntry>) namesystem.getRetryCache().getCacheSet();
|
||||||
assertEquals(22, cacheSet.size());
|
assertEquals(23, cacheSet.size());
|
||||||
|
|
||||||
Map<CacheEntry, CacheEntry> oldEntries =
|
Map<CacheEntry, CacheEntry> oldEntries =
|
||||||
new HashMap<CacheEntry, CacheEntry>();
|
new HashMap<CacheEntry, CacheEntry>();
|
||||||
@ -434,7 +434,7 @@ public class TestNamenodeRetryCache {
|
|||||||
assertTrue(namesystem.hasRetryCache());
|
assertTrue(namesystem.hasRetryCache());
|
||||||
cacheSet = (LightWeightCache<CacheEntry, CacheEntry>) namesystem
|
cacheSet = (LightWeightCache<CacheEntry, CacheEntry>) namesystem
|
||||||
.getRetryCache().getCacheSet();
|
.getRetryCache().getCacheSet();
|
||||||
assertEquals(22, cacheSet.size());
|
assertEquals(23, cacheSet.size());
|
||||||
iter = cacheSet.iterator();
|
iter = cacheSet.iterator();
|
||||||
while (iter.hasNext()) {
|
while (iter.hasNext()) {
|
||||||
CacheEntry entry = iter.next();
|
CacheEntry entry = iter.next();
|
||||||
|
@ -160,7 +160,7 @@ public class TestRetryCacheWithHA {
|
|||||||
FSNamesystem fsn0 = cluster.getNamesystem(0);
|
FSNamesystem fsn0 = cluster.getNamesystem(0);
|
||||||
LightWeightCache<CacheEntry, CacheEntry> cacheSet =
|
LightWeightCache<CacheEntry, CacheEntry> cacheSet =
|
||||||
(LightWeightCache<CacheEntry, CacheEntry>) fsn0.getRetryCache().getCacheSet();
|
(LightWeightCache<CacheEntry, CacheEntry>) fsn0.getRetryCache().getCacheSet();
|
||||||
assertEquals(22, cacheSet.size());
|
assertEquals(23, cacheSet.size());
|
||||||
|
|
||||||
Map<CacheEntry, CacheEntry> oldEntries =
|
Map<CacheEntry, CacheEntry> oldEntries =
|
||||||
new HashMap<CacheEntry, CacheEntry>();
|
new HashMap<CacheEntry, CacheEntry>();
|
||||||
@ -181,7 +181,7 @@ public class TestRetryCacheWithHA {
|
|||||||
FSNamesystem fsn1 = cluster.getNamesystem(1);
|
FSNamesystem fsn1 = cluster.getNamesystem(1);
|
||||||
cacheSet = (LightWeightCache<CacheEntry, CacheEntry>) fsn1
|
cacheSet = (LightWeightCache<CacheEntry, CacheEntry>) fsn1
|
||||||
.getRetryCache().getCacheSet();
|
.getRetryCache().getCacheSet();
|
||||||
assertEquals(22, cacheSet.size());
|
assertEquals(23, cacheSet.size());
|
||||||
iter = cacheSet.iterator();
|
iter = cacheSet.iterator();
|
||||||
while (iter.hasNext()) {
|
while (iter.hasNext()) {
|
||||||
CacheEntry entry = iter.next();
|
CacheEntry entry = iter.next();
|
||||||
@ -1047,6 +1047,49 @@ public class TestRetryCacheWithHA {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** removeXAttr */
|
||||||
|
class RemoveXAttrOp extends AtMostOnceOp {
|
||||||
|
private final String src;
|
||||||
|
|
||||||
|
RemoveXAttrOp(DFSClient client, String src) {
|
||||||
|
super("removeXAttr", client);
|
||||||
|
this.src = src;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
void prepare() throws Exception {
|
||||||
|
Path p = new Path(src);
|
||||||
|
if (!dfs.exists(p)) {
|
||||||
|
DFSTestUtil.createFile(dfs, p, BlockSize, DataNodes, 0);
|
||||||
|
client.setXAttr(src, "user.key", "value".getBytes(),
|
||||||
|
EnumSet.of(XAttrSetFlag.CREATE));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
void invoke() throws Exception {
|
||||||
|
client.removeXAttr(src, "user.key");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
boolean checkNamenodeBeforeReturn() throws Exception {
|
||||||
|
for (int i = 0; i < CHECKTIMES; i++) {
|
||||||
|
Map<String, byte[]> iter = dfs.getXAttrs(new Path(src));
|
||||||
|
Set<String> keySet = iter.keySet();
|
||||||
|
if (!keySet.contains("user.key")) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
Thread.sleep(1000);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
Object getResult() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Test (timeout=60000)
|
@Test (timeout=60000)
|
||||||
public void testCreateSnapshot() throws Exception {
|
public void testCreateSnapshot() throws Exception {
|
||||||
final DFSClient client = genClientWithDummyHandler();
|
final DFSClient client = genClientWithDummyHandler();
|
||||||
@ -1183,6 +1226,13 @@ public class TestRetryCacheWithHA {
|
|||||||
testClientRetryWithFailover(op);
|
testClientRetryWithFailover(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test (timeout=60000)
|
||||||
|
public void testRemoveXAttr() throws Exception {
|
||||||
|
DFSClient client = genClientWithDummyHandler();
|
||||||
|
AtMostOnceOp op = new RemoveXAttrOp(client, "/removexattr");
|
||||||
|
testClientRetryWithFailover(op);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* When NN failover happens, if the client did not receive the response and
|
* When NN failover happens, if the client did not receive the response and
|
||||||
* send a retry request to the other NN, the same response should be recieved
|
* send a retry request to the other NN, the same response should be recieved
|
||||||
|
@ -355,12 +355,6 @@ public class TestParam {
|
|||||||
public void testXAttrNameParam() {
|
public void testXAttrNameParam() {
|
||||||
final XAttrNameParam p = new XAttrNameParam("user.a1");
|
final XAttrNameParam p = new XAttrNameParam("user.a1");
|
||||||
Assert.assertEquals(p.getXAttrName(), "user.a1");
|
Assert.assertEquals(p.getXAttrName(), "user.a1");
|
||||||
try {
|
|
||||||
new XAttrNameParam("a1");
|
|
||||||
Assert.fail();
|
|
||||||
} catch (IllegalArgumentException e) {
|
|
||||||
LOG.info("EXPECTED: " + e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
Binary file not shown.
@ -986,6 +986,8 @@
|
|||||||
<NAMESPACE>USER</NAMESPACE>
|
<NAMESPACE>USER</NAMESPACE>
|
||||||
<NAME>a2</NAME>
|
<NAME>a2</NAME>
|
||||||
</XATTR>
|
</XATTR>
|
||||||
|
<RPC_CLIENTID>e03f4a52-3d85-4e05-8942-286185e639bd</RPC_CLIENTID>
|
||||||
|
<RPC_CALLID>82</RPC_CALLID>
|
||||||
</DATA>
|
</DATA>
|
||||||
</RECORD>
|
</RECORD>
|
||||||
<RECORD>
|
<RECORD>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user