HDFS-420. Fuse-dfs should cache fs handles. Contributed by Brian Bockelman and Eli Collins
git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/trunk@1137675 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
47b416b346
commit
e45025c7ce
|
@ -514,6 +514,8 @@ Trunk (unreleased changes)
|
||||||
|
|
||||||
HDFS-2073. Add @Override annotation to NameNode. (suresh)
|
HDFS-2073. Add @Override annotation to NameNode. (suresh)
|
||||||
|
|
||||||
|
HDFS-420. Fuse-dfs should cache fs handles. (Brian Bockelman and eli)
|
||||||
|
|
||||||
OPTIMIZATIONS
|
OPTIMIZATIONS
|
||||||
|
|
||||||
HDFS-1458. Improve checkpoint performance by avoiding unnecessary image
|
HDFS-1458. Improve checkpoint performance by avoiding unnecessary image
|
||||||
|
|
|
@ -259,7 +259,7 @@ hdfsFS hdfsConnectAsUser(const char* host, tPort port, const char *user)
|
||||||
sprintf(cURI, "hdfs://%s:%d", host, (int)(port));
|
sprintf(cURI, "hdfs://%s:%d", host, (int)(port));
|
||||||
if (cURI == NULL) {
|
if (cURI == NULL) {
|
||||||
fprintf (stderr, "Couldn't allocate an object of size %d",
|
fprintf (stderr, "Couldn't allocate an object of size %d",
|
||||||
strlen(host) + 16);
|
(int)(strlen(host) + 16));
|
||||||
errno = EINTERNAL;
|
errno = EINTERNAL;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,6 +43,8 @@
|
||||||
<property name="test.output" value="no"/>
|
<property name="test.output" value="no"/>
|
||||||
<property name="test.timeout" value="900000"/>
|
<property name="test.timeout" value="900000"/>
|
||||||
<property name="build.dir" location="${hadoop.root}/build/contrib/${name}"/>
|
<property name="build.dir" location="${hadoop.root}/build/contrib/${name}"/>
|
||||||
|
<property name="build.webapps.root.dir" value="${hadoop.root}/build/web"/>
|
||||||
|
<property name="build.webapps" value="${build.webapps.root.dir}/webapps"/>
|
||||||
<property name="build.classes" location="${build.dir}/classes"/>
|
<property name="build.classes" location="${build.dir}/classes"/>
|
||||||
<!-- NB: sun.arch.data.model is not supported on all platforms -->
|
<!-- NB: sun.arch.data.model is not supported on all platforms -->
|
||||||
<property name="build.platform"
|
<property name="build.platform"
|
||||||
|
@ -96,7 +98,8 @@
|
||||||
|
|
||||||
<!-- the unit test classpath -->
|
<!-- the unit test classpath -->
|
||||||
<path id="test.classpath">
|
<path id="test.classpath">
|
||||||
<pathelement location="${build.test}" />
|
<pathelement location="${build.test}"/>
|
||||||
|
<pathelement location="${build.webapps.root.dir}"/>
|
||||||
<pathelement location="${hadoop.root}/build/test/core/classes"/>
|
<pathelement location="${hadoop.root}/build/test/core/classes"/>
|
||||||
<pathelement location="${hadoop.root}/build/test/hdfs/classes"/>
|
<pathelement location="${hadoop.root}/build/test/hdfs/classes"/>
|
||||||
<pathelement location="${hadoop.root}/build/test/mapred/classes"/>
|
<pathelement location="${hadoop.root}/build/test/mapred/classes"/>
|
||||||
|
|
|
@ -28,7 +28,7 @@
|
||||||
<!-- ====================================================== -->
|
<!-- ====================================================== -->
|
||||||
<target name="compile">
|
<target name="compile">
|
||||||
<subant target="compile">
|
<subant target="compile">
|
||||||
<fileset dir="." includes="hdfsproxy/build.xml"/>
|
<fileset dir="." includes="*/build.xml"/>
|
||||||
</subant>
|
</subant>
|
||||||
</target>
|
</target>
|
||||||
|
|
||||||
|
@ -45,6 +45,10 @@
|
||||||
<!-- Test all the contribs. -->
|
<!-- Test all the contribs. -->
|
||||||
<!-- ====================================================== -->
|
<!-- ====================================================== -->
|
||||||
<target name="test">
|
<target name="test">
|
||||||
|
<subant target="test">
|
||||||
|
<fileset dir="." includes="fuse-dfs/build.xml"/>
|
||||||
|
</subant>
|
||||||
|
|
||||||
<!-- hdfsproxy tests failing due to HDFS-1666
|
<!-- hdfsproxy tests failing due to HDFS-1666
|
||||||
<subant target="test">
|
<subant target="test">
|
||||||
<fileset dir="." includes="hdfsproxy/build.xml"/>
|
<fileset dir="." includes="hdfsproxy/build.xml"/>
|
||||||
|
|
|
@ -29,11 +29,7 @@
|
||||||
</fail>
|
</fail>
|
||||||
</target>
|
</target>
|
||||||
|
|
||||||
<target name="compile">
|
<target name="compile" if="fusedfs">
|
||||||
<condition property="perms" value="1" else="0">
|
|
||||||
<not> <isset property="libhdfs.noperms"/> </not>
|
|
||||||
</condition>
|
|
||||||
|
|
||||||
<exec executable="autoreconf" dir="${basedir}"
|
<exec executable="autoreconf" dir="${basedir}"
|
||||||
searchpath="yes" failonerror="yes">
|
searchpath="yes" failonerror="yes">
|
||||||
<arg value="-if"/>
|
<arg value="-if"/>
|
||||||
|
@ -49,7 +45,18 @@
|
||||||
<env key="HADOOP_PREFIX" value="${hadoop.root}"/>
|
<env key="HADOOP_PREFIX" value="${hadoop.root}"/>
|
||||||
<env key="PACKAGE_VERSION" value="0.1.0"/>
|
<env key="PACKAGE_VERSION" value="0.1.0"/>
|
||||||
<env key="BUILD_PLATFORM" value="${build.platform}" />
|
<env key="BUILD_PLATFORM" value="${build.platform}" />
|
||||||
<env key="PERMS" value="${perms}"/>
|
</exec>
|
||||||
|
|
||||||
|
<mkdir dir="${build.dir}"/>
|
||||||
|
<mkdir dir="${build.dir}/test"/>
|
||||||
|
|
||||||
|
<!-- Use exec since the copy task doesn't preserve attrs -->
|
||||||
|
<exec executable="cp" failonerror="true">
|
||||||
|
<arg line="${hadoop.root}/src/contrib/fuse-dfs/src/fuse_dfs ${build.dir}"/>
|
||||||
|
</exec>
|
||||||
|
|
||||||
|
<exec executable="cp" failonerror="true">
|
||||||
|
<arg line="${hadoop.root}/src/contrib/fuse-dfs/src/fuse_dfs_wrapper.sh ${build.dir}"/>
|
||||||
</exec>
|
</exec>
|
||||||
</target>
|
</target>
|
||||||
|
|
||||||
|
@ -57,8 +64,6 @@
|
||||||
<target name="package" />
|
<target name="package" />
|
||||||
|
|
||||||
<target name="compile-test" depends="ivy-retrieve-common, check-libhdfs-exists" if="fusedfs">
|
<target name="compile-test" depends="ivy-retrieve-common, check-libhdfs-exists" if="fusedfs">
|
||||||
<mkdir dir="${build.dir}"/>
|
|
||||||
<mkdir dir="${build.dir}/test"/>
|
|
||||||
<javac encoding="${build.encoding}"
|
<javac encoding="${build.encoding}"
|
||||||
srcdir="${src.test}"
|
srcdir="${src.test}"
|
||||||
includes="**/*.java"
|
includes="**/*.java"
|
||||||
|
@ -66,28 +71,18 @@
|
||||||
debug="${javac.debug}">
|
debug="${javac.debug}">
|
||||||
<classpath refid="test.classpath"/>
|
<classpath refid="test.classpath"/>
|
||||||
</javac>
|
</javac>
|
||||||
|
|
||||||
<!-- Use exec since the copy task doesn't preserve attrs -->
|
|
||||||
<exec executable="cp" failonerror="true">
|
|
||||||
<arg line="${hadoop.root}/src/contrib/fuse-dfs/src/fuse_dfs ${build.dir}"/>
|
|
||||||
</exec>
|
|
||||||
|
|
||||||
<mkdir dir="${build.dir}/test"/>
|
|
||||||
|
|
||||||
<exec executable="cp" failonerror="true">
|
|
||||||
<arg line="${hadoop.root}/src/contrib/fuse-dfs/src/fuse_dfs_wrapper.sh ${build.dir}"/>
|
|
||||||
</exec>
|
|
||||||
</target>
|
</target>
|
||||||
|
|
||||||
<target name="test" depends="compile-test,check-libhdfs-exists" if="fusedfs">
|
<target name="test" depends="compile-test,check-libhdfs-exists" if="fusedfs">
|
||||||
<junit showoutput="${test.output}" fork="yes" printsummary="yes" errorProperty="tests.failed" haltonfailure="no" failureProperty="tests.failed">
|
<junit showoutput="${test.output}" fork="yes" printsummary="yes"
|
||||||
|
errorProperty="tests.failed" haltonfailure="no" failureProperty="tests.failed">
|
||||||
<classpath refid="test.classpath"/>
|
<classpath refid="test.classpath"/>
|
||||||
<sysproperty key="test.build.data" value="${build.test}/data"/>
|
<sysproperty key="test.build.data" value="${build.test}/data"/>
|
||||||
<sysproperty key="build.test" value="${build.test}"/>
|
<sysproperty key="build.test" value="${build.test}"/>
|
||||||
<sysproperty key="user.dir" value="${build.test}/data"/>
|
<sysproperty key="user.dir" value="${build.test}/data"/>
|
||||||
<sysproperty key="hadoop.log.dir" value="${hadoop.log.dir}"/>
|
<sysproperty key="hadoop.log.dir" value="${hadoop.log.dir}"/>
|
||||||
<sysproperty key="test.src.dir" value="${test.src.dir}"/>
|
<sysproperty key="test.src.dir" value="${test.src.dir}"/>
|
||||||
<formatter type="plain" />
|
<formatter type="${test.junit.output.format}" />
|
||||||
<batchtest todir="${build.test}" unless="testcase">
|
<batchtest todir="${build.test}" unless="testcase">
|
||||||
<fileset dir="${src.test}">
|
<fileset dir="${src.test}">
|
||||||
<include name="**/Test*.java"/>
|
<include name="**/Test*.java"/>
|
||||||
|
|
|
@ -42,6 +42,9 @@ AC_PREFIX_DEFAULT([`pwd`])
|
||||||
DEFS=""
|
DEFS=""
|
||||||
AC_SUBST([DEFS])
|
AC_SUBST([DEFS])
|
||||||
|
|
||||||
|
# Need GNU source for multiple hashtables from glibc
|
||||||
|
AC_GNU_SOURCE
|
||||||
|
|
||||||
AC_FUNC_GETGROUPS
|
AC_FUNC_GETGROUPS
|
||||||
AC_TYPE_GETGROUPS
|
AC_TYPE_GETGROUPS
|
||||||
|
|
||||||
|
|
|
@ -17,37 +17,106 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "hdfs.h"
|
#include "hdfs.h"
|
||||||
|
|
||||||
#include "fuse_dfs.h"
|
#include "fuse_dfs.h"
|
||||||
#include "fuse_connect.h"
|
#include "fuse_connect.h"
|
||||||
#include "fuse_users.h"
|
#include "fuse_users.h"
|
||||||
|
|
||||||
|
#include <search.h>
|
||||||
|
|
||||||
#if PERMS
|
#define MAX_ELEMENTS (16 * 1024)
|
||||||
|
static struct hsearch_data *fsTable = NULL;
|
||||||
|
static pthread_mutex_t tableMutex = PTHREAD_MUTEX_INITIALIZER;
|
||||||
|
|
||||||
/**
|
/*
|
||||||
* Connects to the NN as the current user/group according to FUSE
|
* Allocate a hash table for fs handles. Returns 0 on success,
|
||||||
*
|
* -1 on failure.
|
||||||
|
*/
|
||||||
|
int allocFsTable(void) {
|
||||||
|
assert(NULL == fsTable);
|
||||||
|
fsTable = calloc(1, sizeof(struct hsearch_data));
|
||||||
|
if (0 == hcreate_r(MAX_ELEMENTS, fsTable)) {
|
||||||
|
ERROR("Unable to initialize connection table");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Find a fs handle for the given key. Returns a fs handle,
|
||||||
|
* or NULL if there is no fs for the given key.
|
||||||
|
*/
|
||||||
|
static hdfsFS findFs(char *key) {
|
||||||
|
ENTRY entry;
|
||||||
|
ENTRY *entryP = NULL;
|
||||||
|
entry.key = key;
|
||||||
|
if (0 == hsearch_r(entry, FIND, &entryP, fsTable)) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
assert(NULL != entryP->data);
|
||||||
|
return (hdfsFS)entryP->data;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Insert the given fs handle into the table.
|
||||||
|
* Returns 0 on success, -1 on failure.
|
||||||
|
*/
|
||||||
|
static int insertFs(char *key, hdfsFS fs) {
|
||||||
|
ENTRY entry;
|
||||||
|
ENTRY *entryP = NULL;
|
||||||
|
assert(NULL != fs);
|
||||||
|
entry.key = strdup(key);
|
||||||
|
if (entry.key == NULL) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
entry.data = (void*)fs;
|
||||||
|
if (0 == hsearch_r(entry, ENTER, &entryP, fsTable)) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Connect to the NN as the current user/group.
|
||||||
|
* Returns a fs handle on success, or NULL on failure.
|
||||||
*/
|
*/
|
||||||
hdfsFS doConnectAsUser(const char *hostname, int port) {
|
hdfsFS doConnectAsUser(const char *hostname, int port) {
|
||||||
uid_t uid = fuse_get_context()->uid;
|
uid_t uid = fuse_get_context()->uid;
|
||||||
|
|
||||||
char *user = getUsername(uid);
|
char *user = getUsername(uid);
|
||||||
if (NULL == user)
|
int ret;
|
||||||
return NULL;
|
hdfsFS fs = NULL;
|
||||||
int numgroups = 0;
|
if (NULL == user) {
|
||||||
char **groups = getGroups(uid, &numgroups);
|
goto done;
|
||||||
hdfsFS fs = hdfsConnectAsUser(hostname, port, user);
|
}
|
||||||
freeGroups(groups, numgroups);
|
|
||||||
if (user)
|
ret = pthread_mutex_lock(&tableMutex);
|
||||||
|
assert(0 == ret);
|
||||||
|
|
||||||
|
fs = findFs(user);
|
||||||
|
if (NULL == fs) {
|
||||||
|
fs = hdfsConnectAsUserNewInstance(hostname, port, user);
|
||||||
|
if (NULL == fs) {
|
||||||
|
ERROR("Unable to create fs for user %s", user);
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
if (-1 == insertFs(user, fs)) {
|
||||||
|
ERROR("Unable to cache fs for user %s", user);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
done:
|
||||||
|
ret = pthread_mutex_unlock(&tableMutex);
|
||||||
|
assert(0 == ret);
|
||||||
|
if (user) {
|
||||||
free(user);
|
free(user);
|
||||||
|
}
|
||||||
return fs;
|
return fs;
|
||||||
}
|
}
|
||||||
|
|
||||||
#else
|
/*
|
||||||
|
* We currently cache a fs handle per-user in this module rather
|
||||||
hdfsFS doConnectAsUser(const char *hostname, int port) {
|
* than use the FileSystem cache in the java client. Therefore
|
||||||
return hdfsConnect(hostname, port);
|
* we do not disconnect the fs handle here.
|
||||||
|
*/
|
||||||
|
int doDisconnect(hdfsFS fs) {
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
|
@ -21,12 +21,8 @@
|
||||||
|
|
||||||
#include "fuse_dfs.h"
|
#include "fuse_dfs.h"
|
||||||
|
|
||||||
/**
|
hdfsFS doConnectAsUser(const char *hostname, int port);
|
||||||
* Connects to the NN as the current user/group according to FUSE.
|
int doDisconnect(hdfsFS fs);
|
||||||
* Uses the fuse context to get the user name and groups.
|
int allocFsTable(void);
|
||||||
* (if hadoop pre 0.19, will ignore user and group).
|
|
||||||
*/
|
|
||||||
|
|
||||||
hdfsFS doConnectAsUser(const char *hostname, int port) ;
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -33,7 +33,6 @@ typedef struct dfs_context_struct {
|
||||||
int debug;
|
int debug;
|
||||||
char *nn_hostname;
|
char *nn_hostname;
|
||||||
int nn_port;
|
int nn_port;
|
||||||
hdfsFS fs;
|
|
||||||
int read_only;
|
int read_only;
|
||||||
int usetrash;
|
int usetrash;
|
||||||
int direct_io;
|
int direct_io;
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
#include "fuse_options.h"
|
#include "fuse_options.h"
|
||||||
#include "fuse_impls.h"
|
#include "fuse_impls.h"
|
||||||
#include "fuse_init.h"
|
#include "fuse_init.h"
|
||||||
|
#include "fuse_connect.h"
|
||||||
|
|
||||||
int is_protected(const char *path) {
|
int is_protected(const char *path) {
|
||||||
|
|
||||||
|
@ -62,31 +62,25 @@ static struct fuse_operations dfs_oper = {
|
||||||
.truncate = dfs_truncate,
|
.truncate = dfs_truncate,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
{
|
{
|
||||||
|
|
||||||
umask(0);
|
umask(0);
|
||||||
|
|
||||||
extern const char *program;
|
extern const char *program;
|
||||||
program = argv[0];
|
program = argv[0];
|
||||||
struct fuse_args args = FUSE_ARGS_INIT(argc, argv);
|
struct fuse_args args = FUSE_ARGS_INIT(argc, argv);
|
||||||
|
|
||||||
/* clear structure that holds our options */
|
|
||||||
memset(&options, 0, sizeof(struct options));
|
memset(&options, 0, sizeof(struct options));
|
||||||
|
|
||||||
// some defaults
|
|
||||||
options.rdbuffer_size = 10*1024*1024;
|
options.rdbuffer_size = 10*1024*1024;
|
||||||
options.attribute_timeout = 60;
|
options.attribute_timeout = 60;
|
||||||
options.entry_timeout = 60;
|
options.entry_timeout = 60;
|
||||||
|
|
||||||
if (fuse_opt_parse(&args, &options, dfs_opts, dfs_options) == -1)
|
if (-1 == fuse_opt_parse(&args, &options, dfs_opts, dfs_options)) {
|
||||||
/** error parsing options */
|
|
||||||
return -1;
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!options.private) {
|
||||||
// Some fuse options we set
|
|
||||||
if (! options.private) {
|
|
||||||
fuse_opt_add_arg(&args, "-oallow_other");
|
fuse_opt_add_arg(&args, "-oallow_other");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -109,28 +103,24 @@ int main(int argc, char *argv[])
|
||||||
exit(0);
|
exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check connection as root
|
||||||
//
|
|
||||||
// Check we can connect to hdfs
|
|
||||||
//
|
|
||||||
if (options.initchecks == 1) {
|
if (options.initchecks == 1) {
|
||||||
hdfsFS temp;
|
hdfsFS tempFS = hdfsConnectAsUser(options.server, options.port, "root");
|
||||||
if ((temp = hdfsConnect(options.server, options.port)) == NULL) {
|
if (NULL == tempFS) {
|
||||||
const char *cp = getenv("CLASSPATH");
|
const char *cp = getenv("CLASSPATH");
|
||||||
const char *ld = getenv("LD_LIBRARY_PATH");
|
const char *ld = getenv("LD_LIBRARY_PATH");
|
||||||
ERROR("FATAL: misconfiguration - cannot connect to HDFS");
|
ERROR("FATAL: misconfiguration - cannot connect to HDFS");
|
||||||
ERROR("LD_LIBRARY_PATH=%s",ld == NULL ? "NULL" : ld);
|
ERROR("LD_LIBRARY_PATH=%s",ld == NULL ? "NULL" : ld);
|
||||||
ERROR("CLASSPATH=%s",cp == NULL ? "NULL" : cp);
|
ERROR("CLASSPATH=%s",cp == NULL ? "NULL" : cp);
|
||||||
exit(0);
|
exit(1);
|
||||||
|
}
|
||||||
|
if (doDisconnect(tempFS)) {
|
||||||
|
ERROR("FATAL: unable to disconnect from test filesystem.");
|
||||||
|
exit(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int ret = fuse_main(args.argc, args.argv, &dfs_oper, NULL);
|
int ret = fuse_main(args.argc, args.argv, &dfs_oper, NULL);
|
||||||
|
|
||||||
if (ret) printf("\n");
|
|
||||||
|
|
||||||
/** free arguments */
|
|
||||||
fuse_opt_free_args(&args);
|
fuse_opt_free_args(&args);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,11 +35,6 @@
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef linux
|
|
||||||
/* For pread()/pwrite() */
|
|
||||||
#define _XOPEN_SOURCE 500
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef HAVE_SETXATTR
|
#ifdef HAVE_SETXATTR
|
||||||
#include <sys/xattr.h>
|
#include <sys/xattr.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -29,7 +29,7 @@ export JAVA_HOME=/usr/local/java
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ "$LD_LIBRARY_PATH" = "" ]; then
|
if [ "$LD_LIBRARY_PATH" = "" ]; then
|
||||||
export LD_LIBRARY_PATH=$JAVA_HOME/jre/lib/$OS_ARCH/server:/usr/local/share/hdfs/libhdfs/:/usr/local/lib
|
export LD_LIBRARY_PATH=$JAVA_HOME/jre/lib/$OS_ARCH/server:/usr/local/lib
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# If dev build set paths accordingly
|
# If dev build set paths accordingly
|
||||||
|
@ -38,7 +38,7 @@ if [ -d $HADOOP_PREFIX/build ]; then
|
||||||
for f in ${HADOOP_PREFIX}/build/*.jar ; do
|
for f in ${HADOOP_PREFIX}/build/*.jar ; do
|
||||||
export CLASSPATH=$CLASSPATH:$f
|
export CLASSPATH=$CLASSPATH:$f
|
||||||
done
|
done
|
||||||
for f in $HADOOP_PREFIX/build/ivy/lib/Hadoop-Hdfs/common/*.jar ; do
|
for f in $HADOOP_PREFIX/build/ivy/lib/hadoop-hdfs/common/*.jar ; do
|
||||||
export CLASSPATH=$CLASSPATH:$f
|
export CLASSPATH=$CLASSPATH:$f
|
||||||
done
|
done
|
||||||
export PATH=$HADOOP_PREFIX/build/contrib/fuse-dfs:$PATH
|
export PATH=$HADOOP_PREFIX/build/contrib/fuse-dfs:$PATH
|
||||||
|
|
|
@ -23,21 +23,7 @@
|
||||||
int dfs_access(const char *path, int mask)
|
int dfs_access(const char *path, int mask)
|
||||||
{
|
{
|
||||||
TRACE1("access", path)
|
TRACE1("access", path)
|
||||||
// bugbug - I think we need the FileSystemAPI/libhdfs to expose this!
|
assert(path != NULL);
|
||||||
// retrieve dfs specific data
|
// TODO: HDFS-428
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(dfs);
|
|
||||||
assert(path);
|
|
||||||
|
|
||||||
hdfsFS userFS;
|
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port)) == NULL) {
|
|
||||||
ERROR("Could not connect to HDFS");
|
|
||||||
return -EIO;
|
|
||||||
}
|
|
||||||
// return hdfsAccess(userFS, path, mask);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -24,27 +24,30 @@
|
||||||
int dfs_chmod(const char *path, mode_t mode)
|
int dfs_chmod(const char *path, mode_t mode)
|
||||||
{
|
{
|
||||||
TRACE1("chmod", path)
|
TRACE1("chmod", path)
|
||||||
|
int ret = 0;
|
||||||
#if PERMS
|
|
||||||
// retrieve dfs specific data
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
assert('/' == *path);
|
assert('/' == *path);
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect to HDFS");
|
ERROR("Could not connect to HDFS");
|
||||||
return -EIO;
|
ret = -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hdfsChmod(userFS, path, (short)mode)) {
|
if (hdfsChmod(userFS, path, (short)mode)) {
|
||||||
ERROR("Could not chmod %s to %d", path, (int)mode);
|
ERROR("Could not chmod %s to %d", path, (int)mode);
|
||||||
return -EIO;
|
ret = (errno > 0) ? -errno : -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
return 0;
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,10 +26,9 @@
|
||||||
TRACE1("chown", path)
|
TRACE1("chown", path)
|
||||||
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
#if PERMS
|
|
||||||
char *user = NULL;
|
char *user = NULL;
|
||||||
char *group = NULL;
|
char *group = NULL;
|
||||||
|
hdfsFS userFS = NULL;
|
||||||
|
|
||||||
// retrieve dfs specific data
|
// retrieve dfs specific data
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
@ -43,36 +42,40 @@
|
||||||
if (NULL == user) {
|
if (NULL == user) {
|
||||||
ERROR("Could not lookup the user id string %d",(int)uid);
|
ERROR("Could not lookup the user id string %d",(int)uid);
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (0 == ret) {
|
|
||||||
group = getGroup(gid);
|
group = getGroup(gid);
|
||||||
if (group == NULL) {
|
if (group == NULL) {
|
||||||
ERROR("Could not lookup the group id string %d",(int)gid);
|
ERROR("Could not lookup the group id string %d",(int)gid);
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
}
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
hdfsFS userFS = NULL;
|
userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
if (0 == ret) {
|
if (userFS == NULL) {
|
||||||
// if not connected, try to connect and fail out if we can't.
|
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect to HDFS");
|
ERROR("Could not connect to HDFS");
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
}
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (0 == ret) {
|
|
||||||
if (hdfsChown(userFS, path, user, group)) {
|
if (hdfsChown(userFS, path, user, group)) {
|
||||||
ERROR("Could not chown %s to %d:%d", path, (int)uid, gid);
|
ERROR("Could not chown %s to %d:%d", path, (int)uid, gid);
|
||||||
|
ret = (errno > 0) ? -errno : -EIO;
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
cleanup:
|
||||||
|
if (userFS && doDisconnect(userFS)) {
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
}
|
}
|
||||||
}
|
if (user) {
|
||||||
if (user)
|
|
||||||
free(user);
|
free(user);
|
||||||
if (group)
|
}
|
||||||
|
if (group) {
|
||||||
free(group);
|
free(group);
|
||||||
#endif
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,39 +19,36 @@
|
||||||
#include "fuse_dfs.h"
|
#include "fuse_dfs.h"
|
||||||
#include "fuse_impls.h"
|
#include "fuse_impls.h"
|
||||||
#include "fuse_stat_struct.h"
|
#include "fuse_stat_struct.h"
|
||||||
|
#include "fuse_connect.h"
|
||||||
|
|
||||||
int dfs_getattr(const char *path, struct stat *st)
|
int dfs_getattr(const char *path, struct stat *st)
|
||||||
{
|
{
|
||||||
TRACE1("getattr", path)
|
TRACE1("getattr", path)
|
||||||
|
|
||||||
// retrieve dfs specific data
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(st);
|
assert(st);
|
||||||
|
|
||||||
// if not connected, try to connect and fail out if we can't.
|
hdfsFS fs = doConnectAsUser(dfs->nn_hostname,dfs->nn_port);
|
||||||
if (NULL == dfs->fs &&
|
if (NULL == fs) {
|
||||||
NULL == (dfs->fs = hdfsConnect(dfs->nn_hostname,dfs->nn_port))) {
|
|
||||||
ERROR("Could not connect to %s:%d", dfs->nn_hostname, dfs->nn_port);
|
ERROR("Could not connect to %s:%d", dfs->nn_hostname, dfs->nn_port);
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
// call the dfs API to get the actual information
|
int ret = 0;
|
||||||
hdfsFileInfo *info = hdfsGetPathInfo(dfs->fs,path);
|
hdfsFileInfo *info = hdfsGetPathInfo(fs,path);
|
||||||
|
|
||||||
if (NULL == info) {
|
if (NULL == info) {
|
||||||
return -ENOENT;
|
ret = -ENOENT;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
fill_stat_structure(&info[0], st);
|
fill_stat_structure(&info[0], st);
|
||||||
|
|
||||||
// setup hard link info - for a file it is 1 else num entries in a dir + 2 (for . and ..)
|
// setup hard link info - for a file it is 1 else num entries in a dir + 2 (for . and ..)
|
||||||
if (info[0].mKind == kObjectKindDirectory) {
|
if (info[0].mKind == kObjectKindDirectory) {
|
||||||
int numEntries = 0;
|
int numEntries = 0;
|
||||||
hdfsFileInfo *info = hdfsListDirectory(dfs->fs,path,&numEntries);
|
hdfsFileInfo *info = hdfsListDirectory(fs,path,&numEntries);
|
||||||
|
|
||||||
if (info) {
|
if (info) {
|
||||||
hdfsFreeFileInfo(info,numEntries);
|
hdfsFreeFileInfo(info,numEntries);
|
||||||
|
@ -65,5 +62,10 @@ int dfs_getattr(const char *path, struct stat *st)
|
||||||
// free the info pointer
|
// free the info pointer
|
||||||
hdfsFreeFileInfo(info,1);
|
hdfsFreeFileInfo(info,1);
|
||||||
|
|
||||||
return 0;
|
cleanup:
|
||||||
|
if (doDisconnect(fs)) {
|
||||||
|
ERROR("Could not disconnect from filesystem");
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,10 +25,8 @@ int dfs_mkdir(const char *path, mode_t mode)
|
||||||
{
|
{
|
||||||
TRACE1("mkdir", path)
|
TRACE1("mkdir", path)
|
||||||
|
|
||||||
// retrieve dfs specific data
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
assert('/' == *path);
|
assert('/' == *path);
|
||||||
|
@ -43,25 +41,29 @@ int dfs_mkdir(const char *path, mode_t mode)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
}
|
}
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
// In theory the create and chmod should be atomic.
|
// In theory the create and chmod should be atomic.
|
||||||
|
|
||||||
|
int ret = 0;
|
||||||
if (hdfsCreateDirectory(userFS, path)) {
|
if (hdfsCreateDirectory(userFS, path)) {
|
||||||
ERROR("HDFS could not create directory %s", path);
|
ERROR("HDFS could not create directory %s", path);
|
||||||
return -EIO;
|
ret = (errno > 0) ? -errno : -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if PERMS
|
|
||||||
if (hdfsChmod(userFS, path, (short)mode)) {
|
if (hdfsChmod(userFS, path, (short)mode)) {
|
||||||
ERROR("Could not chmod %s to %d", path, (int)mode);
|
ERROR("Could not chmod %s to %d", path, (int)mode);
|
||||||
return -EIO;
|
ret = (errno > 0) ? -errno : -EIO;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
return 0;
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,19 +39,20 @@ int dfs_open(const char *path, struct fuse_file_info *fi)
|
||||||
int flags = (fi->flags & 0x7FFF);
|
int flags = (fi->flags & 0x7FFF);
|
||||||
|
|
||||||
// retrieve dfs specific data
|
// retrieve dfs specific data
|
||||||
dfs_fh *fh = (dfs_fh*)malloc(sizeof (dfs_fh));
|
dfs_fh *fh = (dfs_fh*)calloc(1, sizeof (dfs_fh));
|
||||||
if (fh == NULL) {
|
if (fh == NULL) {
|
||||||
ERROR("Malloc of new file handle failed");
|
ERROR("Malloc of new file handle failed");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((fh->fs = doConnectAsUser(dfs->nn_hostname,dfs->nn_port)) == NULL) {
|
fh->fs = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
|
if (fh->fs == NULL) {
|
||||||
ERROR("Could not connect to dfs");
|
ERROR("Could not connect to dfs");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags & O_RDWR) {
|
if (flags & O_RDWR) {
|
||||||
hdfsFileInfo *info = hdfsGetPathInfo(dfs->fs,path);
|
hdfsFileInfo *info = hdfsGetPathInfo(fh->fs,path);
|
||||||
if (info == NULL) {
|
if (info == NULL) {
|
||||||
// File does not exist (maybe?); interpret it as a O_WRONLY
|
// File does not exist (maybe?); interpret it as a O_WRONLY
|
||||||
// If the actual error was something else, we'll get it again when
|
// If the actual error was something else, we'll get it again when
|
||||||
|
@ -73,28 +74,20 @@ int dfs_open(const char *path, struct fuse_file_info *fi)
|
||||||
return -errno;
|
return -errno;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
|
||||||
// mutex needed for reads/writes
|
|
||||||
//
|
|
||||||
pthread_mutex_init(&fh->mutex, NULL);
|
pthread_mutex_init(&fh->mutex, NULL);
|
||||||
|
|
||||||
if (fi->flags & O_WRONLY || fi->flags & O_CREAT) {
|
if (fi->flags & O_WRONLY || fi->flags & O_CREAT) {
|
||||||
// write specific initialization
|
|
||||||
fh->buf = NULL;
|
fh->buf = NULL;
|
||||||
} else {
|
} else {
|
||||||
// read specific initialization
|
|
||||||
|
|
||||||
assert(dfs->rdbuffer_size > 0);
|
assert(dfs->rdbuffer_size > 0);
|
||||||
|
fh->buf = (char*)malloc(dfs->rdbuffer_size * sizeof(char));
|
||||||
if (NULL == (fh->buf = (char*)malloc(dfs->rdbuffer_size*sizeof (char)))) {
|
if (NULL == fh->buf) {
|
||||||
ERROR("Could not allocate memory for a read for file %s\n", path);
|
ERROR("Could not allocate memory for a read for file %s\n", path);
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
fh->buffersStartOffset = 0;
|
fh->buffersStartOffset = 0;
|
||||||
fh->bufferSize = 0;
|
fh->bufferSize = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
fi->fh = (uint64_t)fh;
|
fi->fh = (uint64_t)fh;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -24,24 +24,15 @@
|
||||||
int dfs_readdir(const char *path, void *buf, fuse_fill_dir_t filler,
|
int dfs_readdir(const char *path, void *buf, fuse_fill_dir_t filler,
|
||||||
off_t offset, struct fuse_file_info *fi)
|
off_t offset, struct fuse_file_info *fi)
|
||||||
{
|
{
|
||||||
TRACE1("readdir",path)
|
TRACE1("readdir", path)
|
||||||
|
|
||||||
(void) offset;
|
|
||||||
(void) fi;
|
|
||||||
|
|
||||||
// retrieve dfs specific data
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(buf);
|
assert(buf);
|
||||||
|
|
||||||
int path_len = strlen(path);
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
|
if (userFS == NULL) {
|
||||||
hdfsFS userFS;
|
|
||||||
// if not connected, try to connect and fail out if we can't.
|
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
@ -50,16 +41,16 @@ int dfs_readdir(const char *path, void *buf, fuse_fill_dir_t filler,
|
||||||
// component (HDFS-975) would save us from parsing it out below.
|
// component (HDFS-975) would save us from parsing it out below.
|
||||||
int numEntries = 0;
|
int numEntries = 0;
|
||||||
hdfsFileInfo *info = hdfsListDirectory(userFS, path, &numEntries);
|
hdfsFileInfo *info = hdfsListDirectory(userFS, path, &numEntries);
|
||||||
userFS = NULL;
|
|
||||||
|
|
||||||
|
int ret = 0;
|
||||||
// NULL means either the directory doesn't exist or maybe IO error.
|
// NULL means either the directory doesn't exist or maybe IO error.
|
||||||
if (NULL == info) {
|
if (NULL == info) {
|
||||||
return -ENOENT;
|
ret = (errno > 0) ? -errno : -ENOENT;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
int i ;
|
int i ;
|
||||||
for (i = 0; i < numEntries; i++) {
|
for (i = 0; i < numEntries; i++) {
|
||||||
|
|
||||||
if (NULL == info[i].mName) {
|
if (NULL == info[i].mName) {
|
||||||
ERROR("Path %s info[%d].mName is NULL", path, i);
|
ERROR("Path %s info[%d].mName is NULL", path, i);
|
||||||
continue;
|
continue;
|
||||||
|
@ -115,5 +106,11 @@ int dfs_readdir(const char *path, void *buf, fuse_fill_dir_t filler,
|
||||||
}
|
}
|
||||||
// free the info pointers
|
// free the info pointers
|
||||||
hdfsFreeFileInfo(info,numEntries);
|
hdfsFreeFileInfo(info,numEntries);
|
||||||
return 0;
|
|
||||||
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
ERROR("Failed to disconnect %d", errno);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
#include "fuse_dfs.h"
|
#include "fuse_dfs.h"
|
||||||
#include "fuse_impls.h"
|
#include "fuse_impls.h"
|
||||||
#include "fuse_file_handle.h"
|
#include "fuse_file_handle.h"
|
||||||
|
#include "fuse_connect.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This mutex is to protect releasing a file handle in case the user calls close in different threads
|
* This mutex is to protect releasing a file handle in case the user calls close in different threads
|
||||||
|
@ -63,6 +64,11 @@ int dfs_release (const char *path, struct fuse_file_info *fi) {
|
||||||
if (fh->buf != NULL) {
|
if (fh->buf != NULL) {
|
||||||
free(fh->buf);
|
free(fh->buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (doDisconnect(fh->fs)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
// this is always created and initialized, so always destroy it. (see dfs_open)
|
// this is always created and initialized, so always destroy it. (see dfs_open)
|
||||||
pthread_mutex_destroy(&fh->mutex);
|
pthread_mutex_destroy(&fh->mutex);
|
||||||
|
|
||||||
|
|
|
@ -46,17 +46,23 @@ int dfs_rename(const char *from, const char *to)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
}
|
}
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int ret = 0;
|
||||||
if (hdfsRename(userFS, from, to)) {
|
if (hdfsRename(userFS, from, to)) {
|
||||||
ERROR("Rename %s to %s failed", from, to);
|
ERROR("Rename %s to %s failed", from, to);
|
||||||
return -EIO;
|
ret = (errno > 0) ? -errno : -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,10 +27,8 @@ int dfs_rmdir(const char *path)
|
||||||
{
|
{
|
||||||
TRACE1("rmdir", path)
|
TRACE1("rmdir", path)
|
||||||
|
|
||||||
// retrieve dfs specific data
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
assert('/' == *path);
|
assert('/' == *path);
|
||||||
|
@ -45,27 +43,34 @@ int dfs_rmdir(const char *path)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
}
|
}
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int ret = 0;
|
||||||
int numEntries = 0;
|
int numEntries = 0;
|
||||||
hdfsFileInfo *info = hdfsListDirectory(userFS,path,&numEntries);
|
hdfsFileInfo *info = hdfsListDirectory(userFS,path,&numEntries);
|
||||||
|
|
||||||
// free the info pointers
|
if (info) {
|
||||||
hdfsFreeFileInfo(info,numEntries);
|
hdfsFreeFileInfo(info, numEntries);
|
||||||
|
}
|
||||||
|
|
||||||
if (numEntries) {
|
if (numEntries) {
|
||||||
return -ENOTEMPTY;
|
ret = -ENOTEMPTY;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hdfsDeleteWithTrash(userFS, path, dfs->usetrash)) {
|
if (hdfsDeleteWithTrash(userFS, path, dfs->usetrash)) {
|
||||||
ERROR("Error trying to delete directory %s", path);
|
ERROR("Error trying to delete directory %s", path);
|
||||||
return -EIO;
|
ret = -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,20 +25,16 @@ int dfs_statfs(const char *path, struct statvfs *st)
|
||||||
{
|
{
|
||||||
TRACE1("statfs",path)
|
TRACE1("statfs",path)
|
||||||
|
|
||||||
// retrieve dfs specific data
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(st);
|
assert(st);
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
|
|
||||||
// init the stat structure
|
|
||||||
memset(st,0,sizeof(struct statvfs));
|
memset(st,0,sizeof(struct statvfs));
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
@ -47,32 +43,15 @@ int dfs_statfs(const char *path, struct statvfs *st)
|
||||||
const tOffset used = hdfsGetUsed(userFS);
|
const tOffset used = hdfsGetUsed(userFS);
|
||||||
const tOffset bsize = hdfsGetDefaultBlockSize(userFS);
|
const tOffset bsize = hdfsGetDefaultBlockSize(userFS);
|
||||||
|
|
||||||
// fill in the statvfs structure
|
if (doDisconnect(userFS)) {
|
||||||
|
return -EIO;
|
||||||
/* FOR REFERENCE:
|
}
|
||||||
struct statvfs {
|
|
||||||
unsigned long f_bsize; // file system block size
|
|
||||||
unsigned long f_frsize; // fragment size
|
|
||||||
fsblkcnt_t f_blocks; // size of fs in f_frsize units
|
|
||||||
fsblkcnt_t f_bfree; // # free blocks
|
|
||||||
fsblkcnt_t f_bavail; // # free blocks for non-root
|
|
||||||
fsfilcnt_t f_files; // # inodes
|
|
||||||
fsfilcnt_t f_ffree; // # free inodes
|
|
||||||
fsfilcnt_t f_favail; // # free inodes for non-root
|
|
||||||
unsigned long f_fsid; // file system id
|
|
||||||
unsigned long f_flag; / mount flags
|
|
||||||
unsigned long f_namemax; // maximum filename length
|
|
||||||
};
|
|
||||||
*/
|
|
||||||
|
|
||||||
st->f_bsize = bsize;
|
st->f_bsize = bsize;
|
||||||
st->f_frsize = bsize;
|
st->f_frsize = bsize;
|
||||||
|
|
||||||
st->f_blocks = cap/bsize;
|
st->f_blocks = cap/bsize;
|
||||||
|
|
||||||
st->f_bfree = (cap-used)/bsize;
|
st->f_bfree = (cap-used)/bsize;
|
||||||
st->f_bavail = (cap-used)/bsize;
|
st->f_bavail = (cap-used)/bsize;
|
||||||
|
|
||||||
st->f_files = 1000;
|
st->f_files = 1000;
|
||||||
st->f_ffree = 500;
|
st->f_ffree = 500;
|
||||||
st->f_favail = 500;
|
st->f_favail = 500;
|
||||||
|
@ -82,4 +61,3 @@ int dfs_statfs(const char *path, struct statvfs *st)
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -29,9 +29,6 @@
|
||||||
int dfs_truncate(const char *path, off_t size)
|
int dfs_truncate(const char *path, off_t size)
|
||||||
{
|
{
|
||||||
TRACE1("truncate", path)
|
TRACE1("truncate", path)
|
||||||
if (size != 0) {
|
|
||||||
return -ENOTSUP;
|
|
||||||
}
|
|
||||||
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
|
@ -39,16 +36,20 @@ int dfs_truncate(const char *path, off_t size)
|
||||||
assert('/' == *path);
|
assert('/' == *path);
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
|
|
||||||
|
if (size != 0) {
|
||||||
|
return -ENOTSUP;
|
||||||
|
}
|
||||||
|
|
||||||
int ret = dfs_unlink(path);
|
int ret = dfs_unlink(path);
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port)) == NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
ret = -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
int flags = O_WRONLY | O_CREAT;
|
int flags = O_WRONLY | O_CREAT;
|
||||||
|
@ -56,12 +57,19 @@ int dfs_truncate(const char *path, off_t size)
|
||||||
hdfsFile file;
|
hdfsFile file;
|
||||||
if ((file = (hdfsFile)hdfsOpenFile(userFS, path, flags, 0, 0, 0)) == NULL) {
|
if ((file = (hdfsFile)hdfsOpenFile(userFS, path, flags, 0, 0, 0)) == NULL) {
|
||||||
ERROR("Could not connect open file %s", path);
|
ERROR("Could not connect open file %s", path);
|
||||||
return -EIO;
|
ret = -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hdfsCloseFile(userFS, file) != 0) {
|
if (hdfsCloseFile(userFS, file) != 0) {
|
||||||
ERROR("Could not close file %s", path);
|
ERROR("Could not close file %s", path);
|
||||||
return -EIO;
|
ret = -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
return 0;
|
|
||||||
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,16 +26,15 @@ int dfs_unlink(const char *path)
|
||||||
{
|
{
|
||||||
TRACE1("unlink", path)
|
TRACE1("unlink", path)
|
||||||
|
|
||||||
// retrieve dfs specific data
|
int ret = 0;
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
assert('/' == *path);
|
assert('/' == *path);
|
||||||
|
|
||||||
if (is_protected(path)) {
|
if (is_protected(path)) {
|
||||||
ERROR("Trying to delete protected directory %s ", path);
|
ERROR("Trying to delete protected directory %s", path);
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,17 +43,22 @@ int dfs_unlink(const char *path)
|
||||||
return -EACCES;
|
return -EACCES;
|
||||||
}
|
}
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hdfsDeleteWithTrash(userFS, path, dfs->usetrash)) {
|
if (hdfsDeleteWithTrash(userFS, path, dfs->usetrash)) {
|
||||||
ERROR("Could not delete file %s", path);
|
ERROR("Could not delete file %s", path);
|
||||||
return -EIO;
|
ret = (errno > 0) ? -errno : -EIO;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,11 +23,9 @@
|
||||||
int dfs_utimens(const char *path, const struct timespec ts[2])
|
int dfs_utimens(const char *path, const struct timespec ts[2])
|
||||||
{
|
{
|
||||||
TRACE1("utimens", path)
|
TRACE1("utimens", path)
|
||||||
#if PERMS
|
int ret = 0;
|
||||||
// retrieve dfs specific data
|
|
||||||
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
dfs_context *dfs = (dfs_context*)fuse_get_context()->private_data;
|
||||||
|
|
||||||
// check params and the context var
|
|
||||||
assert(path);
|
assert(path);
|
||||||
assert(dfs);
|
assert(dfs);
|
||||||
assert('/' == *path);
|
assert('/' == *path);
|
||||||
|
@ -35,25 +33,31 @@ int dfs_utimens(const char *path, const struct timespec ts[2])
|
||||||
time_t aTime = ts[0].tv_sec;
|
time_t aTime = ts[0].tv_sec;
|
||||||
time_t mTime = ts[1].tv_sec;
|
time_t mTime = ts[1].tv_sec;
|
||||||
|
|
||||||
hdfsFS userFS;
|
hdfsFS userFS = doConnectAsUser(dfs->nn_hostname, dfs->nn_port);
|
||||||
// if not connected, try to connect and fail out if we can't.
|
if (userFS == NULL) {
|
||||||
if ((userFS = doConnectAsUser(dfs->nn_hostname,dfs->nn_port))== NULL) {
|
|
||||||
ERROR("Could not connect");
|
ERROR("Could not connect");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hdfsUtime(userFS, path, mTime, aTime)) {
|
if (hdfsUtime(userFS, path, mTime, aTime)) {
|
||||||
hdfsFileInfo *info = hdfsGetPathInfo(dfs->fs,path);
|
hdfsFileInfo *info = hdfsGetPathInfo(userFS, path);
|
||||||
if (info == NULL) {
|
if (info == NULL) {
|
||||||
return -EIO;
|
ret = (errno > 0) ? -errno : -ENOENT;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
// Silently ignore utimens failure for directories, otherwise
|
// Silently ignore utimens failure for directories, otherwise
|
||||||
// some programs like tar will fail.
|
// some programs like tar will fail.
|
||||||
if (info->mKind == kObjectKindDirectory) {
|
if (info->mKind == kObjectKindDirectory) {
|
||||||
return 0;
|
ret = 0;
|
||||||
|
} else {
|
||||||
|
ret = (errno > 0) ? -errno : -EACCES;
|
||||||
}
|
}
|
||||||
return -errno;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
return 0;
|
cleanup:
|
||||||
|
if (doDisconnect(userFS)) {
|
||||||
|
ret = -EIO;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
#include "fuse_init.h"
|
#include "fuse_init.h"
|
||||||
#include "fuse_options.h"
|
#include "fuse_options.h"
|
||||||
#include "fuse_context_handle.h"
|
#include "fuse_context_handle.h"
|
||||||
|
#include "fuse_connect.h"
|
||||||
|
|
||||||
// Hacked up function to basically do:
|
// Hacked up function to basically do:
|
||||||
// protectedpaths = split(options.protected,':');
|
// protectedpaths = split(options.protected,':');
|
||||||
|
@ -30,7 +31,6 @@ void init_protectedpaths(dfs_context *dfs) {
|
||||||
|
|
||||||
char *tmp = options.protected;
|
char *tmp = options.protected;
|
||||||
|
|
||||||
|
|
||||||
// handle degenerate case up front.
|
// handle degenerate case up front.
|
||||||
if (tmp == NULL || 0 == *tmp) {
|
if (tmp == NULL || 0 == *tmp) {
|
||||||
dfs->protectedpaths = (char**)malloc(sizeof(char*));
|
dfs->protectedpaths = (char**)malloc(sizeof(char*));
|
||||||
|
@ -43,7 +43,6 @@ void init_protectedpaths(dfs_context *dfs) {
|
||||||
print_options();
|
print_options();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
while (tmp && (NULL != (tmp = index(tmp,':')))) {
|
while (tmp && (NULL != (tmp = index(tmp,':')))) {
|
||||||
tmp++; // pass the ,
|
tmp++; // pass the ,
|
||||||
|
@ -75,27 +74,15 @@ void init_protectedpaths(dfs_context *dfs) {
|
||||||
j++;
|
j++;
|
||||||
}
|
}
|
||||||
dfs->protectedpaths[j] = NULL;
|
dfs->protectedpaths[j] = NULL;
|
||||||
|
|
||||||
/*
|
|
||||||
j = 0;
|
|
||||||
while (dfs->protectedpaths[j]) {
|
|
||||||
printf("dfs->protectedpaths[%d]=%s\n",j,dfs->protectedpaths[j]);
|
|
||||||
fflush(stdout);
|
|
||||||
j++;
|
|
||||||
}
|
|
||||||
exit(1);
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void *dfs_init()
|
|
||||||
{
|
|
||||||
|
|
||||||
|
void *dfs_init(void) {
|
||||||
//
|
//
|
||||||
// Create a private struct of data we will pass to fuse here and which
|
// Create a private struct of data we will pass to fuse here and which
|
||||||
// will then be accessible on every call.
|
// will then be accessible on every call.
|
||||||
//
|
//
|
||||||
dfs_context *dfs = (dfs_context*)malloc(sizeof (dfs_context));
|
dfs_context *dfs = (dfs_context*)malloc(sizeof(dfs_context));
|
||||||
|
|
||||||
if (NULL == dfs) {
|
if (NULL == dfs) {
|
||||||
ERROR("FATAL: could not malloc dfs_context");
|
ERROR("FATAL: could not malloc dfs_context");
|
||||||
exit(1);
|
exit(1);
|
||||||
|
@ -105,7 +92,6 @@ void *dfs_init()
|
||||||
dfs->debug = options.debug;
|
dfs->debug = options.debug;
|
||||||
dfs->nn_hostname = options.server;
|
dfs->nn_hostname = options.server;
|
||||||
dfs->nn_port = options.port;
|
dfs->nn_port = options.port;
|
||||||
dfs->fs = NULL;
|
|
||||||
dfs->read_only = options.read_only;
|
dfs->read_only = options.read_only;
|
||||||
dfs->usetrash = options.usetrash;
|
dfs->usetrash = options.usetrash;
|
||||||
dfs->protectedpaths = NULL;
|
dfs->protectedpaths = NULL;
|
||||||
|
@ -121,14 +107,17 @@ void *dfs_init()
|
||||||
DEBUG("dfs->rdbuffersize <= 0 = %ld", dfs->rdbuffer_size);
|
DEBUG("dfs->rdbuffersize <= 0 = %ld", dfs->rdbuffer_size);
|
||||||
dfs->rdbuffer_size = 32768;
|
dfs->rdbuffer_size = 32768;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (0 != allocFsTable()) {
|
||||||
|
ERROR("FATAL: could not allocate ");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
return (void*)dfs;
|
return (void*)dfs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void dfs_destroy(void *ptr)
|
||||||
void dfs_destroy (void *ptr)
|
|
||||||
{
|
{
|
||||||
TRACE("destroy")
|
TRACE("destroy")
|
||||||
dfs_context *dfs = (dfs_context*)ptr;
|
|
||||||
dfs->fs = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -107,10 +107,10 @@ int dfs_options(void *data, const char *arg, int key, struct fuse_args *outargs
|
||||||
|
|
||||||
switch (key) {
|
switch (key) {
|
||||||
case FUSE_OPT_KEY_OPT:
|
case FUSE_OPT_KEY_OPT:
|
||||||
fprintf(stderr, "fuse-dfs ignoring option %s\n", arg);
|
INFO("Ignoring option %s", arg);
|
||||||
return 1;
|
return 1;
|
||||||
case KEY_VERSION:
|
case KEY_VERSION:
|
||||||
fprintf(stdout, "%s %s\n", program, _FUSE_DFS_VERSION);
|
INFO("%s %s\n", program, _FUSE_DFS_VERSION);
|
||||||
exit(0);
|
exit(0);
|
||||||
case KEY_HELP:
|
case KEY_HELP:
|
||||||
print_usage(program);
|
print_usage(program);
|
||||||
|
@ -159,14 +159,13 @@ int dfs_options(void *data, const char *arg, int key, struct fuse_args *outargs
|
||||||
} else if (strcmp(arg,"rw") == 0) {
|
} else if (strcmp(arg,"rw") == 0) {
|
||||||
options.read_only = 0;
|
options.read_only = 0;
|
||||||
} else {
|
} else {
|
||||||
ERROR("fuse-dfs didn't recognize %s,%d\n",arg,key);
|
INFO("Adding FUSE arg %s", arg);
|
||||||
fuse_opt_add_arg(outargs,arg);
|
fuse_opt_add_arg(outargs, arg);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
options.port = tmp_port;
|
options.port = tmp_port;
|
||||||
options.server = strdup(tmp_server);
|
options.server = strdup(tmp_server);
|
||||||
ERROR("port=%d,server=%s\n", options.port, options.server);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,16 +25,13 @@
|
||||||
#include "fuse_stat_struct.h"
|
#include "fuse_stat_struct.h"
|
||||||
#include "fuse_context_handle.h"
|
#include "fuse_context_handle.h"
|
||||||
|
|
||||||
#if PERMS
|
/*
|
||||||
/**
|
|
||||||
* getpwuid and getgrgid return static structs so we safeguard the contents
|
* getpwuid and getgrgid return static structs so we safeguard the contents
|
||||||
* while retrieving fields using the 2 structs below.
|
* while retrieving fields using the 2 structs below.
|
||||||
* NOTE: if using both, always get the passwd struct firt!
|
* NOTE: if using both, always get the passwd struct firt!
|
||||||
*/
|
*/
|
||||||
extern pthread_mutex_t passwdstruct_mutex;
|
extern pthread_mutex_t passwdstruct_mutex;
|
||||||
extern pthread_mutex_t groupstruct_mutex;
|
extern pthread_mutex_t groupstruct_mutex;
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
const int default_id = 99; // nobody - not configurable since soon uids in dfs, yeah!
|
const int default_id = 99; // nobody - not configurable since soon uids in dfs, yeah!
|
||||||
const int blksize = 512;
|
const int blksize = 512;
|
||||||
|
@ -55,7 +52,6 @@ int fill_stat_structure(hdfsFileInfo *info, struct stat *st)
|
||||||
st->st_nlink = (info->mKind == kObjectKindDirectory) ? 0 : 1;
|
st->st_nlink = (info->mKind == kObjectKindDirectory) ? 0 : 1;
|
||||||
|
|
||||||
uid_t owner_id = default_id;
|
uid_t owner_id = default_id;
|
||||||
#if PERMS
|
|
||||||
if (info->mOwner != NULL) {
|
if (info->mOwner != NULL) {
|
||||||
//
|
//
|
||||||
// Critical section - protect from concurrent calls in different threads since
|
// Critical section - protect from concurrent calls in different threads since
|
||||||
|
@ -73,9 +69,9 @@ int fill_stat_structure(hdfsFileInfo *info, struct stat *st)
|
||||||
pthread_mutex_unlock(&passwdstruct_mutex);
|
pthread_mutex_unlock(&passwdstruct_mutex);
|
||||||
|
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
gid_t group_id = default_id;
|
gid_t group_id = default_id;
|
||||||
#if PERMS
|
|
||||||
if (info->mGroup != NULL) {
|
if (info->mGroup != NULL) {
|
||||||
//
|
//
|
||||||
// Critical section - protect from concurrent calls in different threads since
|
// Critical section - protect from concurrent calls in different threads since
|
||||||
|
@ -93,15 +89,12 @@ int fill_stat_structure(hdfsFileInfo *info, struct stat *st)
|
||||||
pthread_mutex_unlock(&groupstruct_mutex);
|
pthread_mutex_unlock(&groupstruct_mutex);
|
||||||
|
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
short perm = (info->mKind == kObjectKindDirectory) ? (S_IFDIR | 0777) : (S_IFREG | 0666);
|
short perm = (info->mKind == kObjectKindDirectory) ? (S_IFDIR | 0777) : (S_IFREG | 0666);
|
||||||
#if PERMS
|
|
||||||
if (info->mPermissions > 0) {
|
if (info->mPermissions > 0) {
|
||||||
perm = (info->mKind == kObjectKindDirectory) ? S_IFDIR: S_IFREG ;
|
perm = (info->mKind == kObjectKindDirectory) ? S_IFDIR: S_IFREG ;
|
||||||
perm |= info->mPermissions;
|
perm |= info->mPermissions;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
// set stat metadata
|
// set stat metadata
|
||||||
st->st_size = (info->mKind == kObjectKindDirectory) ? 4096 : info->mSize;
|
st->st_size = (info->mKind == kObjectKindDirectory) ? 4096 : info->mSize;
|
||||||
|
@ -110,11 +103,7 @@ int fill_stat_structure(hdfsFileInfo *info, struct stat *st)
|
||||||
st->st_mode = perm;
|
st->st_mode = perm;
|
||||||
st->st_uid = owner_id;
|
st->st_uid = owner_id;
|
||||||
st->st_gid = group_id;
|
st->st_gid = group_id;
|
||||||
#if PERMS
|
|
||||||
st->st_atime = info->mLastAccess;
|
st->st_atime = info->mLastAccess;
|
||||||
#else
|
|
||||||
st->st_atime = info->mLastMod;
|
|
||||||
#endif
|
|
||||||
st->st_mtime = info->mLastMod;
|
st->st_mtime = info->mLastMod;
|
||||||
st->st_ctime = info->mLastMod;
|
st->st_ctime = info->mLastMod;
|
||||||
|
|
||||||
|
|
|
@ -24,26 +24,19 @@
|
||||||
|
|
||||||
#include "fuse_dfs.h"
|
#include "fuse_dfs.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
#if PERMS
|
|
||||||
/**
|
|
||||||
* getpwuid and getgrgid return static structs so we safeguard the contents
|
* getpwuid and getgrgid return static structs so we safeguard the contents
|
||||||
* while retrieving fields using the 2 structs below.
|
* while retrieving fields using the 2 structs below.
|
||||||
* NOTE: if using both, always get the passwd struct firt!
|
* NOTE: if using both, always get the passwd struct firt!
|
||||||
*/
|
*/
|
||||||
pthread_mutex_t passwdstruct_mutex = PTHREAD_MUTEX_INITIALIZER;
|
pthread_mutex_t passwdstruct_mutex = PTHREAD_MUTEX_INITIALIZER;
|
||||||
pthread_mutex_t groupstruct_mutex = PTHREAD_MUTEX_INITIALIZER;
|
pthread_mutex_t groupstruct_mutex = PTHREAD_MUTEX_INITIALIZER;
|
||||||
#endif
|
|
||||||
|
|
||||||
#if PERMS
|
/*
|
||||||
|
|
||||||
/**
|
|
||||||
* Utility for getting the user making the fuse call in char * form
|
* Utility for getting the user making the fuse call in char * form
|
||||||
* NOTE: if non-null return, the return must be freed by the caller.
|
* NOTE: if non-null return, the return must be freed by the caller.
|
||||||
*/
|
*/
|
||||||
char *getUsername(uid_t uid)
|
char *getUsername(uid_t uid) {
|
||||||
{
|
|
||||||
//
|
//
|
||||||
// Critical section - protect from concurrent calls in different threads.
|
// Critical section - protect from concurrent calls in different threads.
|
||||||
// since the struct below is static.
|
// since the struct below is static.
|
||||||
|
@ -218,4 +211,3 @@ char ** getGroups(uid_t uid, int *num_groups)
|
||||||
#endif
|
#endif
|
||||||
return groupnames;
|
return groupnames;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
Loading…
Reference in New Issue