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:
parent
3ab7f86c16
commit
2e99da4853
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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)
|
||||||
{
|
{
|
||||||
|
|
|
@ -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)
|
||||||
{
|
{
|
||||||
|
|
|
@ -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();
|
||||||
|
|
Loading…
Reference in New Issue