HADOOP-9637. Adding Native Fstat for Windows as needed by YARN. Contributed by Chuan Liu.

git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/trunk@1494341 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Chris Nauroth 2013-06-18 22:15:26 +00:00
parent 3ab7f86c16
commit 2e99da4853
9 changed files with 218 additions and 54 deletions

View File

@ -809,6 +809,9 @@ Release 2.1.0-beta - UNRELEASED
HADOOP-9599. hadoop-config.cmd doesn't set JAVA_LIBRARY_PATH correctly. HADOOP-9599. hadoop-config.cmd doesn't set JAVA_LIBRARY_PATH correctly.
(Mostafa Elhemali via ivanmi) (Mostafa Elhemali via ivanmi)
HADOOP-9637. Adding Native Fstat for Windows as needed by YARN. (Chuan Liu
via cnauroth)
Release 2.0.5-alpha - 06/06/2013 Release 2.0.5-alpha - 06/06/2013
INCOMPATIBLE CHANGES INCOMPATIBLE CHANGES

View File

@ -248,6 +248,20 @@ public class NativeIO {
this.mode = mode; this.mode = mode;
} }
Stat(String owner, String group, int mode) {
if (!Shell.WINDOWS) {
this.owner = owner;
} else {
this.owner = stripDomain(owner);
}
if (!Shell.WINDOWS) {
this.group = group;
} else {
this.group = stripDomain(group);
}
this.mode = mode;
}
@Override @Override
public String toString() { public String toString() {
return "Stat(owner='" + owner + "', group='" + group + "'" + return "Stat(owner='" + owner + "', group='" + group + "'" +
@ -273,9 +287,25 @@ public class NativeIO {
* @throws IOException thrown if there was an IO error while obtaining the file stat. * @throws IOException thrown if there was an IO error while obtaining the file stat.
*/ */
public static Stat getFstat(FileDescriptor fd) throws IOException { public static Stat getFstat(FileDescriptor fd) throws IOException {
Stat stat = fstat(fd); Stat stat = null;
if (!Shell.WINDOWS) {
stat = fstat(fd);
stat.owner = getName(IdCache.USER, stat.ownerId); stat.owner = getName(IdCache.USER, stat.ownerId);
stat.group = getName(IdCache.GROUP, stat.groupId); stat.group = getName(IdCache.GROUP, stat.groupId);
} else {
try {
stat = fstat(fd);
} catch (NativeIOException nioe) {
if (nioe.getErrorCode() == 6) {
throw new NativeIOException("The handle is invalid.",
Errno.EBADF);
} else {
LOG.warn(String.format("NativeIO.getFstat error (%d): %s",
nioe.getErrorCode(), nioe.getMessage()));
throw new NativeIOException("Unknown error", Errno.UNKNOWN);
}
}
}
return stat; return stat;
} }
@ -449,13 +479,26 @@ public class NativeIO {
private static long cacheTimeout; private static long cacheTimeout;
private static boolean initialized = false; private static boolean initialized = false;
/**
* The Windows logon name has two part, NetBIOS domain name and
* user account name, of the format DOMAIN\UserName. This method
* will remove the domain part of the full logon name.
*
* @param the full principal name containing the domain
* @return name with domain removed
*/
private static String stripDomain(String name) {
int i = name.indexOf('\\');
if (i != -1)
name = name.substring(i + 1);
return name;
}
public static String getOwner(FileDescriptor fd) throws IOException { public static String getOwner(FileDescriptor fd) throws IOException {
ensureInitialized(); ensureInitialized();
if (Shell.WINDOWS) { if (Shell.WINDOWS) {
String owner = Windows.getOwner(fd); String owner = Windows.getOwner(fd);
int i = owner.indexOf('\\'); owner = stripDomain(owner);
if (i != -1)
owner = owner.substring(i + 1);
return owner; return owner;
} else { } else {
long uid = POSIX.getUIDforFDOwnerforOwner(fd); long uid = POSIX.getUIDforFDOwnerforOwner(fd);

View File

@ -50,6 +50,7 @@
// the NativeIO$POSIX$Stat inner class and its constructor // the NativeIO$POSIX$Stat inner class and its constructor
static jclass stat_clazz; static jclass stat_clazz;
static jmethodID stat_ctor; static jmethodID stat_ctor;
static jmethodID stat_ctor2;
// the NativeIOException class and its constructor // the NativeIOException class and its constructor
static jclass nioe_clazz; static jclass nioe_clazz;
@ -84,10 +85,12 @@ static int workaround_non_threadsafe_calls(JNIEnv *env, jclass clazz) {
return result; return result;
} }
#ifdef UNIX
static void stat_init(JNIEnv *env, jclass nativeio_class) { static void stat_init(JNIEnv *env, jclass nativeio_class) {
jclass clazz = NULL;
jclass obj_class = NULL;
jmethodID obj_ctor = NULL;
// Init Stat // Init Stat
jclass clazz = (*env)->FindClass(env, "org/apache/hadoop/io/nativeio/NativeIO$POSIX$Stat"); clazz = (*env)->FindClass(env, "org/apache/hadoop/io/nativeio/NativeIO$POSIX$Stat");
if (!clazz) { if (!clazz) {
return; // exception has been raised return; // exception has been raised
} }
@ -100,12 +103,16 @@ static void stat_init(JNIEnv *env, jclass nativeio_class) {
if (!stat_ctor) { if (!stat_ctor) {
return; // exception has been raised return; // exception has been raised
} }
stat_ctor2 = (*env)->GetMethodID(env, stat_clazz, "<init>",
jclass obj_class = (*env)->FindClass(env, "java/lang/Object"); "(Ljava/lang/String;Ljava/lang/String;I)V");
if (!stat_ctor2) {
return; // exception has been raised
}
obj_class = (*env)->FindClass(env, "java/lang/Object");
if (!obj_class) { if (!obj_class) {
return; // exception has been raised return; // exception has been raised
} }
jmethodID obj_ctor = (*env)->GetMethodID(env, obj_class, obj_ctor = (*env)->GetMethodID(env, obj_class,
"<init>", "()V"); "<init>", "()V");
if (!obj_ctor) { if (!obj_ctor) {
return; // exception has been raised return; // exception has been raised
@ -130,7 +137,6 @@ static void stat_deinit(JNIEnv *env) {
pw_lock_object = NULL; pw_lock_object = NULL;
} }
} }
#endif
static void nioe_init(JNIEnv *env) { static void nioe_init(JNIEnv *env) {
// Init NativeIOException // Init NativeIOException
@ -168,10 +174,8 @@ static void nioe_deinit(JNIEnv *env) {
JNIEXPORT void JNICALL JNIEXPORT void JNICALL
Java_org_apache_hadoop_io_nativeio_NativeIO_initNative( Java_org_apache_hadoop_io_nativeio_NativeIO_initNative(
JNIEnv *env, jclass clazz) { JNIEnv *env, jclass clazz) {
#ifdef UNIX
stat_init(env, clazz); stat_init(env, clazz);
PASS_EXCEPTIONS_GOTO(env, error); PASS_EXCEPTIONS_GOTO(env, error);
#endif
nioe_init(env); nioe_init(env);
PASS_EXCEPTIONS_GOTO(env, error); PASS_EXCEPTIONS_GOTO(env, error);
fd_init(env); fd_init(env);
@ -229,9 +233,44 @@ cleanup:
#endif #endif
#ifdef WINDOWS #ifdef WINDOWS
THROW(env, "java/io/IOException", LPWSTR owner = NULL;
"The function POSIX.fstat() is not supported on Windows"); LPWSTR group = NULL;
return NULL; int mode;
jstring jstr_owner = NULL;
jstring jstr_group = NULL;
int rc;
jobject ret = NULL;
HANDLE hFile = (HANDLE) fd_get(env, fd_object);
PASS_EXCEPTIONS_GOTO(env, cleanup);
rc = FindFileOwnerAndPermissionByHandle(hFile, &owner, &group, &mode);
if (rc != ERROR_SUCCESS) {
throw_ioe(env, rc);
goto cleanup;
}
jstr_owner = (*env)->NewString(env, owner, (jsize) wcslen(owner));
if (jstr_owner == NULL) goto cleanup;
jstr_group = (*env)->NewString(env, group, (jsize) wcslen(group));;
if (jstr_group == NULL) goto cleanup;
ret = (*env)->NewObject(env, stat_clazz, stat_ctor2,
jstr_owner, jstr_group, (jint)mode);
cleanup:
if (ret == NULL) {
if (jstr_owner != NULL)
(*env)->ReleaseStringChars(env, jstr_owner, owner);
if (jstr_group != NULL)
(*env)->ReleaseStringChars(env, jstr_group, group);
}
LocalFree(owner);
LocalFree(group);
return ret;
#endif #endif
} }

View File

@ -561,22 +561,11 @@ static BOOL ConvertActionsToMask(__in LPCWSTR path,
{ {
MODE_CHANGE_ACTION const *curr = NULL; MODE_CHANGE_ACTION const *curr = NULL;
BY_HANDLE_FILE_INFORMATION fileInformation;
DWORD dwErrorCode = ERROR_SUCCESS; DWORD dwErrorCode = ERROR_SUCCESS;
INT mode = 0; INT mode = 0;
dwErrorCode = GetFileInformationByName(path, FALSE, &fileInformation); dwErrorCode = FindFileOwnerAndPermission(path, FALSE, NULL, NULL, &mode);
if (dwErrorCode != ERROR_SUCCESS)
{
ReportErrorCode(L"GetFileInformationByName", dwErrorCode);
return FALSE;
}
if (IsDirFileInfo(&fileInformation))
{
mode |= UX_DIRECTORY;
}
dwErrorCode = FindFileOwnerAndPermission(path, NULL, NULL, &mode);
if (dwErrorCode != ERROR_SUCCESS) if (dwErrorCode != ERROR_SUCCESS)
{ {
ReportErrorCode(L"FindFileOwnerAndPermission", dwErrorCode); ReportErrorCode(L"FindFileOwnerAndPermission", dwErrorCode);

View File

@ -52,7 +52,7 @@ static DWORD ChangeFileOwnerBySid(__in LPCWSTR path,
// Get a pointer to the existing owner information and DACL // Get a pointer to the existing owner information and DACL
// //
dwRtnCode = FindFileOwnerAndPermission(longPathName, NULL, NULL, &oldMode); dwRtnCode = FindFileOwnerAndPermission(longPathName, FALSE, NULL, NULL, &oldMode);
if (dwRtnCode != ERROR_SUCCESS) if (dwRtnCode != ERROR_SUCCESS)
{ {
goto ChangeFileOwnerByNameEnd; goto ChangeFileOwnerByNameEnd;

View File

@ -63,6 +63,7 @@ enum UnixAclMask
UX_U_WRITE = 00200, // S_IWUSR UX_U_WRITE = 00200, // S_IWUSR
UX_U_READ = 00400, // S_IRUSR UX_U_READ = 00400, // S_IRUSR
UX_DIRECTORY = 0040000, // S_IFDIR UX_DIRECTORY = 0040000, // S_IFDIR
UX_REGULAR = 0100000, // S_IFREG
UX_SYMLINK = 0120000, // S_IFLNK UX_SYMLINK = 0120000, // S_IFLNK
}; };
@ -130,6 +131,13 @@ BOOL IsDirFileInfo(const BY_HANDLE_FILE_INFORMATION *fileInformation);
DWORD FindFileOwnerAndPermission( DWORD FindFileOwnerAndPermission(
__in LPCWSTR pathName, __in LPCWSTR pathName,
__in BOOL followLink,
__out_opt LPWSTR *pOwnerName,
__out_opt LPWSTR *pGroupName,
__out_opt PINT pMask);
DWORD FindFileOwnerAndPermissionByHandle(
__in HANDLE fileHandle,
__out_opt LPWSTR *pOwnerName, __out_opt LPWSTR *pOwnerName,
__out_opt LPWSTR *pGroupName, __out_opt LPWSTR *pGroupName,
__out_opt PINT pMask); __out_opt PINT pMask);

View File

@ -707,6 +707,71 @@ CheckAccessEnd:
return dwRtnCode; return dwRtnCode;
} }
//----------------------------------------------------------------------------
// Function: FindFileOwnerAndPermissionByHandle
//
// Description:
// Find the owner, primary group and permissions of a file object given the
// the file object handle. The function will always follow symbolic links.
//
// Returns:
// ERROR_SUCCESS: on success
// Error code otherwise
//
// Notes:
// - Caller needs to destroy the memeory of owner and group names by calling
// LocalFree() function.
//
// - If the user or group name does not exist, the user or group SID will be
// returned as the name.
//
DWORD FindFileOwnerAndPermissionByHandle(
__in HANDLE fileHandle,
__out_opt LPWSTR *pOwnerName,
__out_opt LPWSTR *pGroupName,
__out_opt PINT pMask)
{
LPWSTR path = NULL;
DWORD cchPathLen = 0;
DWORD dwRtnCode = ERROR_SUCCESS;
DWORD ret = ERROR_SUCCESS;
dwRtnCode = GetFinalPathNameByHandle(fileHandle, path, cchPathLen, 0);
if (dwRtnCode == 0)
{
ret = GetLastError();
goto FindFileOwnerAndPermissionByHandleEnd;
}
cchPathLen = dwRtnCode;
path = (LPWSTR) LocalAlloc(LPTR, cchPathLen * sizeof(WCHAR));
if (path == NULL)
{
ret = GetLastError();
goto FindFileOwnerAndPermissionByHandleEnd;
}
dwRtnCode = GetFinalPathNameByHandle(fileHandle, path, cchPathLen, 0);
if (dwRtnCode != cchPathLen - 1)
{
ret = GetLastError();
goto FindFileOwnerAndPermissionByHandleEnd;
}
dwRtnCode = FindFileOwnerAndPermission(path, TRUE, pOwnerName, pGroupName, pMask);
if (dwRtnCode != ERROR_SUCCESS)
{
ret = dwRtnCode;
goto FindFileOwnerAndPermissionByHandleEnd;
}
FindFileOwnerAndPermissionByHandleEnd:
LocalFree(path);
return ret;
}
//---------------------------------------------------------------------------- //----------------------------------------------------------------------------
// Function: FindFileOwnerAndPermission // Function: FindFileOwnerAndPermission
// //
@ -726,6 +791,7 @@ CheckAccessEnd:
// //
DWORD FindFileOwnerAndPermission( DWORD FindFileOwnerAndPermission(
__in LPCWSTR pathName, __in LPCWSTR pathName,
__in BOOL followLink,
__out_opt LPWSTR *pOwnerName, __out_opt LPWSTR *pOwnerName,
__out_opt LPWSTR *pGroupName, __out_opt LPWSTR *pGroupName,
__out_opt PINT pMask) __out_opt PINT pMask)
@ -740,6 +806,9 @@ DWORD FindFileOwnerAndPermission(
DWORD cbSid = SECURITY_MAX_SID_SIZE; DWORD cbSid = SECURITY_MAX_SID_SIZE;
PACL pDacl = NULL; PACL pDacl = NULL;
BOOL isSymlink;
BY_HANDLE_FILE_INFORMATION fileInformation;
ACCESS_MASK ownerAccessRights = 0; ACCESS_MASK ownerAccessRights = 0;
ACCESS_MASK groupAccessRights = 0; ACCESS_MASK groupAccessRights = 0;
ACCESS_MASK worldAccessRights = 0; ACCESS_MASK worldAccessRights = 0;
@ -801,6 +870,28 @@ DWORD FindFileOwnerAndPermission(
if (pMask == NULL) goto FindFileOwnerAndPermissionEnd; if (pMask == NULL) goto FindFileOwnerAndPermissionEnd;
dwRtnCode = GetFileInformationByName(pathName,
followLink, &fileInformation);
if (dwRtnCode != ERROR_SUCCESS)
{
ret = dwRtnCode;
goto FindFileOwnerAndPermissionEnd;
}
dwRtnCode = SymbolicLinkCheck(pathName, &isSymlink);
if (dwRtnCode != ERROR_SUCCESS)
{
ret = dwRtnCode;
goto FindFileOwnerAndPermissionEnd;
}
if (isSymlink)
*pMask |= UX_SYMLINK;
else if (IsDirFileInfo(&fileInformation))
*pMask |= UX_DIRECTORY;
else
*pMask |= UX_REGULAR;
if ((dwRtnCode = GetEffectiveRightsForSid(pSd, if ((dwRtnCode = GetEffectiveRightsForSid(pSd,
psidOwner, &ownerAccessRights)) != ERROR_SUCCESS) psidOwner, &ownerAccessRights)) != ERROR_SUCCESS)
{ {

View File

@ -253,8 +253,6 @@ int Ls(__in int argc, __in_ecount(argc) wchar_t *argv[])
LARGE_INTEGER fileSize; LARGE_INTEGER fileSize;
BOOL isSymlink = FALSE;
int ret = EXIT_FAILURE; int ret = EXIT_FAILURE;
int optionsMask = 0; int optionsMask = 0;
@ -290,19 +288,8 @@ int Ls(__in int argc, __in_ecount(argc) wchar_t *argv[])
goto LsEnd; goto LsEnd;
} }
dwErrorCode = SymbolicLinkCheck(longPathName, &isSymlink);
if (dwErrorCode != ERROR_SUCCESS)
{
ReportErrorCode(L"IsSymbolicLink", dwErrorCode);
goto LsEnd;
}
if (isSymlink)
unixAccessMode |= UX_SYMLINK;
else if (IsDirFileInfo(&fileInformation))
unixAccessMode |= UX_DIRECTORY;
dwErrorCode = FindFileOwnerAndPermission(longPathName, dwErrorCode = FindFileOwnerAndPermission(longPathName,
optionsMask & CmdLineOptionFollowSymlink,
&ownerName, &groupName, &unixAccessMode); &ownerName, &groupName, &unixAccessMode);
if (dwErrorCode != ERROR_SUCCESS) if (dwErrorCode != ERROR_SUCCESS)
{ {

View File

@ -26,6 +26,7 @@ import java.io.FileWriter;
import java.io.IOException; import java.io.IOException;
import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.atomic.AtomicReference;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.List; import java.util.List;
import org.junit.Assert; import org.junit.Assert;
@ -42,6 +43,7 @@ import org.apache.hadoop.fs.FileUtil;
import org.apache.hadoop.fs.FileSystem; import org.apache.hadoop.fs.FileSystem;
import org.apache.hadoop.fs.Path; import org.apache.hadoop.fs.Path;
import org.apache.hadoop.fs.permission.FsPermission; import org.apache.hadoop.fs.permission.FsPermission;
import org.apache.hadoop.security.UserGroupInformation;
import org.apache.hadoop.util.NativeCodeLoader; import org.apache.hadoop.util.NativeCodeLoader;
import org.apache.hadoop.util.Time; import org.apache.hadoop.util.Time;
@ -64,17 +66,23 @@ public class TestNativeIO {
@Test (timeout = 30000) @Test (timeout = 30000)
public void testFstat() throws Exception { public void testFstat() throws Exception {
if (Path.WINDOWS) {
return;
}
FileOutputStream fos = new FileOutputStream( FileOutputStream fos = new FileOutputStream(
new File(TEST_DIR, "testfstat")); new File(TEST_DIR, "testfstat"));
NativeIO.POSIX.Stat stat = NativeIO.POSIX.getFstat(fos.getFD()); NativeIO.POSIX.Stat stat = NativeIO.POSIX.getFstat(fos.getFD());
fos.close(); fos.close();
LOG.info("Stat: " + String.valueOf(stat)); LOG.info("Stat: " + String.valueOf(stat));
assertEquals(System.getProperty("user.name"), stat.getOwner()); String owner = stat.getOwner();
String expectedOwner = System.getProperty("user.name");
if (Path.WINDOWS) {
UserGroupInformation ugi =
UserGroupInformation.createRemoteUser(expectedOwner);
final String adminsGroupString = "Administrators";
if (Arrays.asList(ugi.getGroupNames()).contains(adminsGroupString)) {
expectedOwner = adminsGroupString;
}
}
assertEquals(expectedOwner, owner);
assertNotNull(stat.getGroup()); assertNotNull(stat.getGroup());
assertTrue(!stat.getGroup().isEmpty()); assertTrue(!stat.getGroup().isEmpty());
assertEquals("Stat mode field should indicate a regular file", assertEquals("Stat mode field should indicate a regular file",
@ -136,10 +144,6 @@ public class TestNativeIO {
@Test (timeout = 30000) @Test (timeout = 30000)
public void testFstatClosedFd() throws Exception { public void testFstatClosedFd() throws Exception {
if (Path.WINDOWS) {
return;
}
FileOutputStream fos = new FileOutputStream( FileOutputStream fos = new FileOutputStream(
new File(TEST_DIR, "testfstat2")); new File(TEST_DIR, "testfstat2"));
fos.close(); fos.close();