+
+enum {
+ KEY_HELP,
+ KEY_VERSION,
+};
+
+
+static struct fuse_opt xattrs_opts[] = {
+ FUSE_OPT_KEY("-V", KEY_VERSION),
+ FUSE_OPT_KEY("--version", KEY_VERSION),
+ FUSE_OPT_KEY("-h", KEY_HELP),
+ FUSE_OPT_KEY("--help", KEY_HELP),
+ FUSE_OPT_END
+};
+
+int is_directory(const char *path) {
+ struct stat statbuf;
+ if (stat(path, &statbuf) != 0) {
+ fprintf(stderr, "cannot get source directory status: %s\n", path);
+ return -1;
+ }
+
+ if (!S_ISDIR(statbuf.st_mode)) {
+ fprintf(stderr, "source directory must be a directory: %s\n", path);
+ return -1;
+ }
+
+ return 1;
+}
+
+/**
+ * Check if the path is valid. If it's a relative path,
+ * prepend the working path.
+ * @param path relative or absolute path to eval.
+ * @return new string with absolute path
+ */
+const char *sanitized_source_directory(const char *path) {
+ char *absolute_path;
+ if (strlen(path) == 0) {
+ return NULL;
+ }
+
+ /* absolute path, we don't do anything */
+ if (path[0] == '/') {
+ if (is_directory(path) == -1) {
+ return NULL;
+ }
+ absolute_path = strdup(path);
+ return absolute_path;
+ }
+
+ char *pwd = get_current_dir_name();
+ size_t len = strlen(pwd) + 1 + strlen(path) + 1;
+ int has_trailing_backslash = (path[strlen(path)-1] == '/');
+ if (!has_trailing_backslash)
+ len++;
+
+ absolute_path = (char*) malloc(sizeof(char) * len);
+ memset(absolute_path, '\0', len);
+ sprintf(absolute_path, "%s/%s", pwd, path);
+
+ if(!has_trailing_backslash)
+ absolute_path[len-2] = '/';
+
+ if (is_directory(absolute_path) == -1) {
+ free(absolute_path);
+ return NULL;
+ }
+
+ return absolute_path;
+}
+
+static int xattrs_opt_proc(void *data, const char *arg, int key,
+ struct fuse_args *outargs) {
+ (void) data;
+ switch (key) {
+ case FUSE_OPT_KEY_NONOPT:
+ if (!xattrs_config.source_dir) {
+ xattrs_config.source_dir = sanitized_source_directory(arg);
+ return 0;
+ }
+ break;
+
+ case KEY_HELP:
+ fprintf(stderr,
+ "usage: %s source_dir mountpoint [options]\n"
+ "\n"
+ "general options:\n"
+ " -o opt,[opt...] mount options\n"
+ " -h --help print help\n"
+ " -V --version print version\n"
+ "\n"
+ "FUSE XATTRS options:\n"
+ "\n", outargs->argv[0]);
+
+ fuse_opt_add_arg(outargs, "-ho");
+ fuse_main(outargs->argc, outargs->argv, &xmp_oper, NULL);
+ exit(1);
+
+ case KEY_VERSION:
+ fuse_opt_add_arg(outargs, "--version");
+ fuse_main(outargs->argc, outargs->argv, &xmp_oper, NULL);
+ exit(0);
+ }
+ return 1;
+}
+
+
+
+int main(int argc, char *argv[]) {
+ struct fuse_args args = FUSE_ARGS_INIT(argc, argv);
+ if (fuse_opt_parse(&args, &xattrs_config, xattrs_opts, xattrs_opt_proc) == -1) {
+ exit(1);
+ }
+
+ if (!xattrs_config.source_dir) {
+ fprintf(stderr, "missing source directory\n");
+ fprintf(stderr, "see `%s -h' for usage\n", argv[0]);
+ exit(1);
+ }
+