diff options
| -rw-r--r-- | metaentry.c | 68 | ||||
| -rw-r--r-- | metastore.c | 139 | ||||
| -rw-r--r-- | utils.c | 19 | ||||
| -rw-r--r-- | utils.h | 4 | 
4 files changed, 125 insertions, 105 deletions
| diff --git a/metaentry.c b/metaentry.c index 3af4a6c..83f84fa 100644 --- a/metaentry.c +++ b/metaentry.c @@ -19,6 +19,7 @@  #include "metaentry.h"  #include "utils.h" +#if 0  static void  mentry_free(struct metaentry *m)  { @@ -42,6 +43,7 @@ mentry_free(struct metaentry *m)  	free(m);  } +#endif  static struct metaentry *  mentry_alloc() @@ -86,33 +88,34 @@ mentry_insert(struct metaentry *mentry, struct metaentry **mhead)  	prev->next = mentry;  } +#ifdef DEBUG  static void  mentry_print(const struct metaentry *mentry)  {  	int i;  	if (!mentry || !mentry->path) { -		fprintf(stderr, "Incorrect meta entry passed to printmetaentry\n"); +		msg(MSG_DEBUG, "Incorrect meta entry passed to printmetaentry\n");  		return;  	} -	printf("===========================\n"); -	printf("Dump of metaentry %p\n", mentry); -	printf("===========================\n"); +	msg(MSG_DEBUG, "===========================\n"); +	msg(MSG_DEBUG, "Dump of metaentry %p\n", mentry); +	msg(MSG_DEBUG, "===========================\n"); -	printf("path\t\t: %s\n", mentry->path); -	printf("owner\t\t: %s\n", mentry->owner); -	printf("group\t\t: %s\n", mentry->group); -	printf("mtime\t\t: %ld\n", (unsigned long)mentry->mtime); -	printf("mtimensec\t: %ld\n", (unsigned long)mentry->mtimensec); -	printf("mode\t\t: %ld\n", (unsigned long)mentry->mode); +	msg(MSG_DEBUG, "path\t\t: %s\n", mentry->path); +	msg(MSG_DEBUG, "owner\t\t: %s\n", mentry->owner); +	msg(MSG_DEBUG, "group\t\t: %s\n", mentry->group); +	msg(MSG_DEBUG, "mtime\t\t: %ld\n", (unsigned long)mentry->mtime); +	msg(MSG_DEBUG, "mtimensec\t: %ld\n", (unsigned long)mentry->mtimensec); +	msg(MSG_DEBUG, "mode\t\t: %ld\n", (unsigned long)mentry->mode);  	for (i = 0; i < mentry->xattrs; i++) { -		printf("xattr[%i]\t: %s=\"", i, mentry->xattr_names[i]); +		msg(MSG_DEBUG, "xattr[%i]\t: %s=\"", i, mentry->xattr_names[i]);  		binary_print(mentry->xattr_values[i], mentry->xattr_lvalues[i]); -		printf("\"\n"); +		msg(MSG_DEBUG, "\"\n");  	} -	printf("===========================\n\n"); +	msg(MSG_DEBUG, "===========================\n\n");  }  static void @@ -126,8 +129,9 @@ mentries_print(const struct metaentry *mhead)  		mentry_print(mentry);  	} -	printf("%i entries in total\n", i); +	msg(MSG_DEBUG, "%i entries in total\n", i);  } +#endif  static struct metaentry *  mentry_create(const char *path) @@ -141,19 +145,19 @@ mentry_create(const char *path)  	struct metaentry *mentry;  	if (lstat(path, &sbuf)) { -		perror("lstat"); +		msg(MSG_ERROR, "lstat failed for %s: %s\n", path, strerror(errno));  		return NULL;  	}  	pbuf = getpwuid(sbuf.st_uid);  	if (!pbuf) { -		perror("getpwuid"); +		msg(MSG_ERROR, "getpwuid failed for %i: %s\n", (int)sbuf.st_uid, strerror(errno));  		return NULL;  	}  	gbuf = getgrgid(sbuf.st_gid);  	if (!gbuf) { -		perror("getgrgid"); +		msg(MSG_ERROR, "getgrgid failed for %i: %s\n", (int)sbuf.st_gid, strerror(errno));  		return NULL;  	} @@ -171,14 +175,14 @@ mentry_create(const char *path)  	lsize = listxattr(path, NULL, 0);  	if (lsize < 0) { -		perror("listxattr"); +		msg(MSG_ERROR, "listxattr failed for %s: %s\n", path, strerror(errno));  		return NULL;  	}  	list = xmalloc(lsize);  	lsize = listxattr(path, list, lsize);  	if (lsize < 0) { -		perror("listxattr"); +		msg(MSG_ERROR, "listxattr failed for %s: %s\n", path, strerror(errno));  		return NULL;  	} @@ -205,7 +209,7 @@ mentry_create(const char *path)  		mentry->xattr_names[i] = xstrdup(attr);  		vsize = getxattr(path, attr, NULL, 0);  		if (vsize < 0) { -			perror("getxattr"); +			msg(MSG_ERROR, "getxattr failed for %s: %s\n", path, strerror(errno));  			return NULL;  		} @@ -214,7 +218,7 @@ mentry_create(const char *path)  		vsize = getxattr(path, attr, mentry->xattr_values[i], vsize);  		if (vsize < 0) { -			perror("getxattr"); +			msg(MSG_ERROR, "getxattr failed for %s: %s\n", path, strerror(errno));  			return NULL;  		}  		i++; @@ -260,22 +264,20 @@ mentries_recurse(const char *path, struct metaentry **mhead)  		return;  	if (lstat(path, &sbuf)) { -		printf("Failed to stat %s\n", path); +		msg(MSG_ERROR, "lstat failed for %s: %s\n", path, strerror(errno));  		return;  	}  	mentry = mentry_create(path); -	if (!mentry) { -		printf("Failed to get metadata for %s\n", path); +	if (!mentry)  		return; -	}  	mentry_insert(mentry, mhead);  	if (S_ISDIR(sbuf.st_mode)) {  		dir = opendir(path);  		if (!dir) { -			printf("Failed to open dir %s\n", path); +			msg(MSG_ERROR, "opendir failed for %s: %s\n", path, strerror(errno));  			return;  		} @@ -356,33 +358,33 @@ mentries_fromfile(struct metaentry **mhead, const char *path)  	}  	if (sbuf.st_size < (SIGNATURELEN + VERSIONLEN)) { -		fprintf(stderr, "Invalid size for file %s\n", path); +		msg(MSG_CRITICAL, "File %s has an invalid size\n", path);  		exit(EXIT_FAILURE);  	}  	mmapstart = mmap(NULL, (size_t)sbuf.st_size, PROT_READ, MAP_SHARED, fd, 0);  	if (mmapstart == MAP_FAILED) { -		perror("mmap"); +		msg(MSG_CRITICAL, "Unable to mmap %s: %s\n", path, strerror(errno));  		exit(EXIT_FAILURE);  	}  	ptr = mmapstart;  	max = mmapstart + sbuf.st_size;  	if (strncmp(ptr, SIGNATURE, SIGNATURELEN)) { -		printf("Invalid signature for file %s\n", path); +		msg(MSG_CRITICAL, "Invalid signature for file %s\n", path);  		goto out;  	}  	ptr += SIGNATURELEN;  	if (strncmp(ptr, VERSION, VERSIONLEN)) { -		printf("Invalid version for file %s\n", path); +		msg(MSG_CRITICAL, "Invalid version of file %s\n", path);  		goto out;  	}  	ptr += VERSIONLEN;  	while (ptr < mmapstart + sbuf.st_size) {  		if (*ptr == '\0') { -			fprintf(stderr, "Invalid characters in file %s\n", path); +			msg(MSG_CRITICAL, "Invalid characters in file %s\n", path);  			goto out;  		} @@ -471,7 +473,7 @@ mentry_compare(struct metaentry *left, struct metaentry *right)  	int retval = DIFF_NONE;  	if (!left || !right) { -		fprintf(stderr, "mentry_compare called with empty arguments\n"); +		msg(MSG_ERROR, "%s called with empty list\n", __FUNCTION__);  		return -1;  	} @@ -511,7 +513,7 @@ mentries_compare(struct metaentry *mheadleft,  	int cmp;  	if (!mheadleft || !mheadright) { -		fprintf(stderr, "mentries_compare called with empty list\n"); +		msg(MSG_ERROR, "%s called with empty list\n", __FUNCTION__);  		return;  	} diff --git a/metastore.c b/metastore.c index ec28c38..7a2ac7a 100644 --- a/metastore.c +++ b/metastore.c @@ -1,4 +1,3 @@ -#include <stdio.h>  #include <sys/types.h>  #include <sys/stat.h>  #include <pwd.h> @@ -20,34 +19,34 @@ static void  compare_print(struct metaentry *left, struct metaentry *right, int cmp)  {  	if (!left) { -		printf("Path %s: removed\n", right->path); +		msg(MSG_QUIET, "%s:\tremoved\n", right->path);  		return;  	}  	if (!right) { -		printf("Path %s: added\n", left->path); +		msg(MSG_QUIET, "%s:\tadded\n", left->path);  		return;  	}  	if (cmp == DIFF_NONE) { -		msg(MSG_DEBUG, "Path %s: no difference\n", left->path); +		msg(MSG_DEBUG, "%s:\tno difference\n", left->path);  		return;  	} -	printf("Path %s: ", left->path); +	msg(MSG_QUIET, "%s:\t", left->path);  	if (cmp & DIFF_OWNER) -		printf("owner "); +		msg(MSG_QUIET, "owner ");  	if (cmp & DIFF_GROUP) -		printf("group "); +		msg(MSG_QUIET, "group ");  	if (cmp & DIFF_MODE) -		printf("mode "); +		msg(MSG_QUIET, "mode ");  	if (cmp & DIFF_TYPE) -		printf("type "); +		msg(MSG_QUIET, "type ");  	if (cmp & DIFF_MTIME) -		printf("mtime "); +		msg(MSG_QUIET, "mtime ");  	if (cmp & DIFF_XATTR) -		printf("xattr "); -	printf("\n"); +		msg(MSG_QUIET, "xattr "); +	msg(MSG_QUIET, "\n");  }  static void @@ -61,73 +60,73 @@ compare_fix(struct metaentry *left, struct metaentry *right, int cmp)  	int i;  	if (!left && !right) { -		printf("%s called with incorrect arguments\n", __FUNCTION__); +		msg(MSG_ERROR, "%s called with incorrect arguments\n", __FUNCTION__);  		return;  	}  	if (!left) { -		printf("Path %s: removed\n", right->path); +		msg(MSG_NORMAL, "%s:\tremoved\n", right->path);  		return;  	}  	if (!right) { -		printf("Path %s: added\n", left->path); +		msg(MSG_NORMAL, "%s:\tadded\n", left->path);  		return;  	}  	if (cmp == DIFF_NONE) { -		msg(MSG_DEBUG, "Path %s: no difference\n", left->path); +		msg(MSG_DEBUG, "%s:\tno difference\n", left->path);  		return;  	}  	if (cmp & DIFF_TYPE) { -		printf("Path %s: new type, will not change metadata\n", left->path); +		msg(MSG_NORMAL, "%s:\tnew type, will not change metadata\n", left->path);  		return;  	} -	if (cmp & (DIFF_OWNER | DIFF_GROUP)) { +	msg(MSG_QUIET, "%s:\tchanging metadata\n", left->path); + +	while (cmp & (DIFF_OWNER | DIFF_GROUP)) {  		if (cmp & DIFF_OWNER) { -			printf("Path %s: fixing owner from %s to %s\n", left->path, left->group, right->group); +			msg(MSG_NORMAL, "\tchanging owner from %s to %s\n", left->path, left->group, right->group);  			owner = getpwnam(right->owner);  			if (!owner) { -				perror("getpwnam"); -				return; +				msg(MSG_DEBUG, "\tgetpwnam failed: %s\n", strerror(errno)); +				break;  			}  			uid = owner->pw_uid;  		}  		if (cmp & DIFF_GROUP) { -			printf("Path %s: fixing group from %s to %s\n", left->path, left->group, right->group); +			msg(MSG_NORMAL, "\tchanging group from %s to %s\n", left->path, left->group, right->group);  			group = getgrnam(right->group);  			if (!group) { -				perror("getgrnam"); -				return; +				msg(MSG_DEBUG, "\tgetgrnam failed: %s\n", strerror(errno)); +				break;  			}  			gid = group->gr_gid;  		}  		if (lchown(left->path, uid, gid)) { -			perror("lchown"); -			return; +			msg(MSG_DEBUG, "\tlchown failed: %s\n", strerror(errno)); +			break;  		} -		printf("Success\n"); +		break;  	}  	if (cmp & DIFF_MODE) { -		printf("Path %s: fixing mode from 0%o to 0%o\n", left->path, left->mode, right->mode); -		if (chmod(left->path, left->mode)) { -			perror("chmod"); -			return; -		} +		msg(MSG_NORMAL, "%s:\tchanging mode from 0%o to 0%o\n", left->path, left->mode, right->mode); +		if (chmod(left->path, left->mode)) +			msg(MSG_DEBUG, "\tchmod failed: %s\n", strerror(errno));  	}  	if (cmp & DIFF_MTIME) { -		printf("Path %s: fixing mtime %ld to %ld\n", left->path, left->mtime, right->mtime); +		msg(MSG_NORMAL, "%s:\tchanging mtime from %ld to %ld\n", left->path, left->mtime, right->mtime);  		/* FIXME: Use utimensat here */  		tbuf.actime = right->mtime;  		tbuf.modtime = right->mtime;  		if (utime(left->path, &tbuf)) { -			perror("utime"); +			msg(MSG_DEBUG, "\tutime failed: %s\n", strerror(errno));  			return;  		}  	} @@ -138,10 +137,10 @@ compare_fix(struct metaentry *left, struct metaentry *right, int cmp)  			if (mentry_find_xattr(right, left, i) >= 0)  				continue; -			msg(MSG_NORMAL, "Path %s: removing xattr %s\n", +			msg(MSG_NORMAL, "%s:\tremoving xattr %s\n",  			    left->path, left->xattr_names[i]);  			if (lremovexattr(left->path, left->xattr_names[i])) -				perror("lremovexattr"); +				msg(MSG_DEBUG, "\tlremovexattr failed: %s\n", strerror(errno));  		}  		for (i = 0; i < right->xattrs; i++) { @@ -149,31 +148,31 @@ compare_fix(struct metaentry *left, struct metaentry *right, int cmp)  			if (mentry_find_xattr(left, right, i) >= 0)  				continue; -			msg(MSG_NORMAL, "Path %s: adding xattr %s\n", +			msg(MSG_NORMAL, "%s:\tadding xattr %s\n",  			    right->path, right->xattr_names[i]);  			if (lsetxattr(right->path, right->xattr_names[i],  -				       right->xattr_values[i], right->xattr_lvalues[i], XATTR_CREATE)) -				perror("lsetxattr"); +				      right->xattr_values[i], right->xattr_lvalues[i], XATTR_CREATE)) +				msg(MSG_DEBUG, "\tlsetxattr failed: %s\n", strerror(errno));  		}  	}  }  static void -usage(const char *arg0, const char *msg) +usage(const char *arg0, const char *message)  { -	if (msg) -		fprintf(stderr, "%s: %s\n\n", arg0, msg); -	fprintf(stderr, "Usage: %s ACTION [OPTIONS] [PATH]...\n\n", arg0); -	fprintf(stderr, "Where ACTION is one of:\n" -			"  -d, --diff\tShow differences between stored and actual metadata\n" -			"  -s, --save\tSave current metadata\n" -			"  -a, --apply\tApply stored metadata\n" -			"  -h, --help\tHelp message (this text)\n\n" -			"Valid OPTIONS are (can be given more than once):\n" -			"  -v, --verbose\tPrint more verbose messages\n" -			"  -q, --quiet\tPrint less verbose messages\n"); - -	exit(msg ? EXIT_FAILURE : EXIT_SUCCESS); +	if (message) +		msg(MSG_CRITICAL, "%s: %s\n\n", arg0, msg); +	msg(MSG_CRITICAL, "Usage: %s ACTION [OPTIONS] [PATH]...\n\n", arg0); +	msg(MSG_CRITICAL, "Where ACTION is one of:\n" +	    "  -d, --diff\tShow differences between stored and actual metadata\n" +	    "  -s, --save\tSave current metadata\n" +	    "  -a, --apply\tApply stored metadata\n" +	    "  -h, --help\tHelp message (this text)\n\n" +	    "Valid OPTIONS are (can be given more than once):\n" +	    "  -v, --verbose\tPrint more verbose messages\n" +	    "  -q, --quiet\tPrint less verbose messages\n"); + +	exit(message ? EXIT_FAILURE : EXIT_SUCCESS);  }  static struct option long_options[] = { @@ -251,51 +250,63 @@ main(int argc, char **argv, char **envp)  	case ACTION_DIFF:  		mentries_fromfile(&mfhead, METAFILE);  		if (!mfhead) { -			fprintf(stderr, "Failed to load metadata from file\n"); +			msg(MSG_CRITICAL, "Failed to load metadata from file\n");  			exit(EXIT_FAILURE);  		} -		if (optind < argc) +		if (optind < argc) {  			while (optind < argc)  				mentries_recurse_path(argv[optind++], &mhead); -		else +		} else {  			mentries_recurse_path(".", &mhead); +		} +  		if (!mhead) { -			fprintf(stderr, "Failed to load metadata from fs\n"); +			msg(MSG_CRITICAL, "Failed to load metadata from file system\n");  			exit(EXIT_FAILURE);  		} +  		mentries_compare(mhead, mfhead, compare_print);  		break; +  	case ACTION_SAVE: -		if (optind < argc) +		if (optind < argc) {  			while (optind < argc)  				mentries_recurse_path(argv[optind++], &mhead); -		else +		} else {  			mentries_recurse_path(".", &mhead); +		} +  		if (!mhead) { -			fprintf(stderr, "Failed to load metadata from fs\n"); +			msg(MSG_CRITICAL, "Failed to load metadata from file system\n");  			exit(EXIT_FAILURE);  		} +  		mentries_tofile(mhead, METAFILE);  		break; +  	case ACTION_APPLY:  		mentries_fromfile(&mfhead, METAFILE);  		if (!mfhead) { -			fprintf(stderr, "Failed to load metadata from file\n"); +			msg(MSG_CRITICAL, "Failed to load metadata from file\n");  			exit(EXIT_FAILURE);  		} -		if (optind < argc) +		if (optind < argc) {  			while (optind < argc)  				mentries_recurse_path(argv[optind++], &mhead); -		else +		} else {  			mentries_recurse_path(".", &mhead); +		} +  		if (!mhead) { -			fprintf(stderr, "Failed to load metadata from fs\n"); +			msg(MSG_CRITICAL, "Failed to load metadata from file system\n");  			exit(EXIT_FAILURE);  		} +  		mentries_compare(mhead, mfhead, compare_fix);  		break; +  	case ACTION_HELP:  		usage(argv[0], NULL);  	} @@ -19,7 +19,12 @@ msg(int level, const char *fmt, ...)  		return 0;  	va_start(ap, fmt); -	ret = vfprintf(stderr, fmt, ap); + +	if (level < MSG_QUIET) +		ret = vfprintf(stderr, fmt, ap); +	else +		ret = vfprintf(stdout, fmt, ap); +  	va_end(ap);  	return ret;  } @@ -29,7 +34,7 @@ xmalloc(size_t size)  {          void *result = malloc(size);          if (!result) { -                fprintf(stderr, "Failed to malloc %zi bytes\n", size); +                msg(MSG_CRITICAL, "Failed to malloc %zi bytes\n", size);                  exit(EXIT_FAILURE);          }          return result; @@ -40,7 +45,7 @@ xstrdup(const char *s)  {  	char *result = strdup(s);  	if (!result) { -		fprintf(stderr, "Failed to strdup %zi bytes\n", strlen(s)); +		msg(MSG_CRITICAL, "Failed to strdup %zi bytes\n", strlen(s));  		exit(EXIT_FAILURE);  	}  	return result; @@ -53,9 +58,9 @@ binary_print(const char *s, ssize_t len)  	for (i = 0; i < len; i++) {  		if (isprint(s[i])) -			printf("%c", s[i]); +			msg(MSG_DEBUG, "%c", s[i]);  		else -			printf("0x%02X", (int)s[i]); +			msg(MSG_DEBUG, "0x%02X", (int)s[i]);  	}  } @@ -98,7 +103,7 @@ read_int(char **from, size_t len, const char *max)  	int i;  	if (*from + len > max) { -		fprintf(stderr, "Attempt to read beyond end of file, corrupt file?\n"); +		msg(MSG_CRITICAL, "Attempt to read beyond end of file, corrupt file?\n");  		exit(EXIT_FAILURE);  	} @@ -114,7 +119,7 @@ read_binary_string(char **from, size_t len, const char *max)  	char *result;  	if (*from + len > max) { -		fprintf(stderr, "Attempt to read beyond end of file, corrupt file?\n"); +		msg(MSG_CRITICAL, "Attempt to read beyond end of file, corrupt file?\n");  		exit(EXIT_FAILURE);  	} @@ -4,7 +4,9 @@ extern int verbosity;  #define MSG_NORMAL    0  #define MSG_DEBUG     1  #define MSG_QUIET    -1 -#define MSG_CRITICAL -2 +#define MSG_ERROR    -2 +#define MSG_CRITICAL -3 +  int msg(int level, const char *fmt, ...);  void *xmalloc(size_t size); | 
