Blame examples/diff.c

Packit Service 20376f
/*
Packit Service 20376f
 * libgit2 "diff" example - shows how to use the diff API
Packit Service 20376f
 *
Packit Service 20376f
 * Written by the libgit2 contributors
Packit Service 20376f
 *
Packit Service 20376f
 * To the extent possible under law, the author(s) have dedicated all copyright
Packit Service 20376f
 * and related and neighboring rights to this software to the public domain
Packit Service 20376f
 * worldwide. This software is distributed without any warranty.
Packit Service 20376f
 *
Packit Service 20376f
 * You should have received a copy of the CC0 Public Domain Dedication along
Packit Service 20376f
 * with this software. If not, see
Packit Service 20376f
 * <http://creativecommons.org/publicdomain/zero/1.0/>.
Packit Service 20376f
 */
Packit Service 20376f
Packit Service 20376f
#include "common.h"
Packit Service 20376f
Packit Service 20376f
/**
Packit Service 20376f
 * This example demonstrates the use of the libgit2 diff APIs to
Packit Service 20376f
 * create `git_diff` objects and display them, emulating a number of
Packit Service 20376f
 * core Git `diff` command line options.
Packit Service 20376f
 *
Packit Service 20376f
 * This covers on a portion of the core Git diff options and doesn't
Packit Service 20376f
 * have particularly good error handling, but it should show most of
Packit Service 20376f
 * the core libgit2 diff APIs, including various types of diffs and
Packit Service 20376f
 * how to do renaming detection and patch formatting.
Packit Service 20376f
 */
Packit Service 20376f
Packit Service 20376f
static const char *colors[] = {
Packit Service 20376f
	"\033[m", /* reset */
Packit Service 20376f
	"\033[1m", /* bold */
Packit Service 20376f
	"\033[31m", /* red */
Packit Service 20376f
	"\033[32m", /* green */
Packit Service 20376f
	"\033[36m" /* cyan */
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
enum {
Packit Service 20376f
	OUTPUT_DIFF = (1 << 0),
Packit Service 20376f
	OUTPUT_STAT = (1 << 1),
Packit Service 20376f
	OUTPUT_SHORTSTAT = (1 << 2),
Packit Service 20376f
	OUTPUT_NUMSTAT = (1 << 3),
Packit Service 20376f
	OUTPUT_SUMMARY = (1 << 4)
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
enum {
Packit Service 20376f
	CACHE_NORMAL = 0,
Packit Service 20376f
	CACHE_ONLY = 1,
Packit Service 20376f
	CACHE_NONE = 2
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
/** The 'opts' struct captures all the various parsed command line options. */
Packit Service 20376f
struct opts {
Packit Service 20376f
	git_diff_options diffopts;
Packit Service 20376f
	git_diff_find_options findopts;
Packit Service 20376f
	int color;
Packit Service 20376f
	int cache;
Packit Service 20376f
	int output;
Packit Service 20376f
	git_diff_format_t format;
Packit Service 20376f
	const char *treeish1;
Packit Service 20376f
	const char *treeish2;
Packit Service 20376f
	const char *dir;
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
/** These functions are implemented at the end */
Packit Service 20376f
static void usage(const char *message, const char *arg);
Packit Service 20376f
static void parse_opts(struct opts *o, int argc, char *argv[]);
Packit Service 20376f
static int color_printer(
Packit Service 20376f
	const git_diff_delta*, const git_diff_hunk*, const git_diff_line*, void*);
Packit Service 20376f
static void diff_print_stats(git_diff *diff, struct opts *o);
Packit Service 20376f
Packit Service 20376f
int main(int argc, char *argv[])
Packit Service 20376f
{
Packit Service 20376f
	git_repository *repo = NULL;
Packit Service 20376f
	git_tree *t1 = NULL, *t2 = NULL;
Packit Service 20376f
	git_diff *diff;
Packit Service 20376f
	struct opts o = {
Packit Service 20376f
		GIT_DIFF_OPTIONS_INIT, GIT_DIFF_FIND_OPTIONS_INIT,
Packit Service 20376f
		-1, 0, 0, GIT_DIFF_FORMAT_PATCH, NULL, NULL, "."
Packit Service 20376f
	};
Packit Service 20376f
Packit Service 20376f
	git_libgit2_init();
Packit Service 20376f
Packit Service 20376f
	parse_opts(&o, argc, argv);
Packit Service 20376f
Packit Service 20376f
	check_lg2(git_repository_open_ext(&repo, o.dir, 0, NULL),
Packit Service 20376f
		"Could not open repository", o.dir);
Packit Service 20376f
Packit Service 20376f
	/**
Packit Service 20376f
	 * Possible argument patterns:
Packit Service 20376f
	 *
Packit Service 20376f
	 *  * <sha1> <sha2>
Packit Service 20376f
	 *  * <sha1> --cached
Packit Service 20376f
	 *  * <sha1>
Packit Service 20376f
	 *  * --cached
Packit Service 20376f
	 *  * --nocache (don't use index data in diff at all)
Packit Service 20376f
	 *  * nothing
Packit Service 20376f
	 *
Packit Service 20376f
	 * Currently ranged arguments like <sha1>..<sha2> and <sha1>...<sha2>
Packit Service 20376f
	 * are not supported in this example
Packit Service 20376f
	 */
Packit Service 20376f
Packit Service 20376f
	if (o.treeish1)
Packit Service 20376f
		treeish_to_tree(&t1, repo, o.treeish1);
Packit Service 20376f
	if (o.treeish2)
Packit Service 20376f
		treeish_to_tree(&t2, repo, o.treeish2);
Packit Service 20376f
Packit Service 20376f
	if (t1 && t2)
Packit Service 20376f
		check_lg2(
Packit Service 20376f
			git_diff_tree_to_tree(&diff, repo, t1, t2, &o.diffopts),
Packit Service 20376f
			"diff trees", NULL);
Packit Service 20376f
	else if (o.cache != CACHE_NORMAL) {
Packit Service 20376f
		if (!t1)
Packit Service 20376f
			treeish_to_tree(&t1, repo, "HEAD");
Packit Service 20376f
Packit Service 20376f
		if (o.cache == CACHE_NONE)
Packit Service 20376f
			check_lg2(
Packit Service 20376f
				git_diff_tree_to_workdir(&diff, repo, t1, &o.diffopts),
Packit Service 20376f
				"diff tree to working directory", NULL);
Packit Service 20376f
		else
Packit Service 20376f
			check_lg2(
Packit Service 20376f
				git_diff_tree_to_index(&diff, repo, t1, NULL, &o.diffopts),
Packit Service 20376f
				"diff tree to index", NULL);
Packit Service 20376f
	}
Packit Service 20376f
	else if (t1)
Packit Service 20376f
		check_lg2(
Packit Service 20376f
			git_diff_tree_to_workdir_with_index(&diff, repo, t1, &o.diffopts),
Packit Service 20376f
			"diff tree to working directory", NULL);
Packit Service 20376f
	else
Packit Service 20376f
		check_lg2(
Packit Service 20376f
			git_diff_index_to_workdir(&diff, repo, NULL, &o.diffopts),
Packit Service 20376f
			"diff index to working directory", NULL);
Packit Service 20376f
Packit Service 20376f
	/** Apply rename and copy detection if requested. */
Packit Service 20376f
Packit Service 20376f
	if ((o.findopts.flags & GIT_DIFF_FIND_ALL) != 0)
Packit Service 20376f
		check_lg2(
Packit Service 20376f
			git_diff_find_similar(diff, &o.findopts),
Packit Service 20376f
			"finding renames and copies", NULL);
Packit Service 20376f
Packit Service 20376f
	/** Generate simple output using libgit2 display helper. */
Packit Service 20376f
Packit Service 20376f
	if (!o.output)
Packit Service 20376f
		o.output = OUTPUT_DIFF;
Packit Service 20376f
Packit Service 20376f
	if (o.output != OUTPUT_DIFF)
Packit Service 20376f
		diff_print_stats(diff, &o);
Packit Service 20376f
Packit Service 20376f
	if ((o.output & OUTPUT_DIFF) != 0) {
Packit Service 20376f
		if (o.color >= 0)
Packit Service 20376f
			fputs(colors[0], stdout);
Packit Service 20376f
Packit Service 20376f
		check_lg2(
Packit Service 20376f
			git_diff_print(diff, o.format, color_printer, &o.color),
Packit Service 20376f
			"displaying diff", NULL);
Packit Service 20376f
Packit Service 20376f
		if (o.color >= 0)
Packit Service 20376f
			fputs(colors[0], stdout);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/** Cleanup before exiting. */
Packit Service 20376f
Packit Service 20376f
	git_diff_free(diff);
Packit Service 20376f
	git_tree_free(t1);
Packit Service 20376f
	git_tree_free(t2);
Packit Service 20376f
	git_repository_free(repo);
Packit Service 20376f
Packit Service 20376f
	git_libgit2_shutdown();
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void usage(const char *message, const char *arg)
Packit Service 20376f
{
Packit Service 20376f
	if (message && arg)
Packit Service 20376f
		fprintf(stderr, "%s: %s\n", message, arg);
Packit Service 20376f
	else if (message)
Packit Service 20376f
		fprintf(stderr, "%s\n", message);
Packit Service 20376f
	fprintf(stderr, "usage: diff [<tree-oid> [<tree-oid>]]\n");
Packit Service 20376f
	exit(1);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/** This implements very rudimentary colorized output. */
Packit Service 20376f
static int color_printer(
Packit Service 20376f
	const git_diff_delta *delta,
Packit Service 20376f
	const git_diff_hunk *hunk,
Packit Service 20376f
	const git_diff_line *line,
Packit Service 20376f
	void *data)
Packit Service 20376f
{
Packit Service 20376f
	int *last_color = data, color = 0;
Packit Service 20376f
Packit Service 20376f
	(void)delta; (void)hunk;
Packit Service 20376f
Packit Service 20376f
	if (*last_color >= 0) {
Packit Service 20376f
		switch (line->origin) {
Packit Service 20376f
		case GIT_DIFF_LINE_ADDITION:  color = 3; break;
Packit Service 20376f
		case GIT_DIFF_LINE_DELETION:  color = 2; break;
Packit Service 20376f
		case GIT_DIFF_LINE_ADD_EOFNL: color = 3; break;
Packit Service 20376f
		case GIT_DIFF_LINE_DEL_EOFNL: color = 2; break;
Packit Service 20376f
		case GIT_DIFF_LINE_FILE_HDR:  color = 1; break;
Packit Service 20376f
		case GIT_DIFF_LINE_HUNK_HDR:  color = 4; break;
Packit Service 20376f
		default: break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (color != *last_color) {
Packit Service 20376f
			if (*last_color == 1 || color == 1)
Packit Service 20376f
				fputs(colors[0], stdout);
Packit Service 20376f
			fputs(colors[color], stdout);
Packit Service 20376f
			*last_color = color;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return diff_output(delta, hunk, line, stdout);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/** Parse arguments as copied from git-diff. */
Packit Service 20376f
static void parse_opts(struct opts *o, int argc, char *argv[])
Packit Service 20376f
{
Packit Service 20376f
	struct args_info args = ARGS_INFO_INIT;
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
	for (args.pos = 1; args.pos < argc; ++args.pos) {
Packit Service 20376f
		const char *a = argv[args.pos];
Packit Service 20376f
Packit Service 20376f
		if (a[0] != '-') {
Packit Service 20376f
			if (o->treeish1 == NULL)
Packit Service 20376f
				o->treeish1 = a;
Packit Service 20376f
			else if (o->treeish2 == NULL)
Packit Service 20376f
				o->treeish2 = a;
Packit Service 20376f
			else
Packit Service 20376f
				usage("Only one or two tree identifiers can be provided", NULL);
Packit Service 20376f
		}
Packit Service 20376f
		else if (!strcmp(a, "-p") || !strcmp(a, "-u") ||
Packit Service 20376f
				 !strcmp(a, "--patch")) {
Packit Service 20376f
			o->output |= OUTPUT_DIFF;
Packit Service 20376f
			o->format = GIT_DIFF_FORMAT_PATCH;
Packit Service 20376f
		}
Packit Service 20376f
		else if (!strcmp(a, "--cached"))
Packit Service 20376f
			o->cache = CACHE_ONLY;
Packit Service 20376f
		else if (!strcmp(a, "--nocache"))
Packit Service 20376f
			o->cache = CACHE_NONE;
Packit Service 20376f
		else if (!strcmp(a, "--name-only") || !strcmp(a, "--format=name"))
Packit Service 20376f
			o->format = GIT_DIFF_FORMAT_NAME_ONLY;
Packit Service 20376f
		else if (!strcmp(a, "--name-status") ||
Packit Service 20376f
				!strcmp(a, "--format=name-status"))
Packit Service 20376f
			o->format = GIT_DIFF_FORMAT_NAME_STATUS;
Packit Service 20376f
		else if (!strcmp(a, "--raw") || !strcmp(a, "--format=raw"))
Packit Service 20376f
			o->format = GIT_DIFF_FORMAT_RAW;
Packit Service 20376f
		else if (!strcmp(a, "--format=diff-index")) {
Packit Service 20376f
			o->format = GIT_DIFF_FORMAT_RAW;
Packit Service 20376f
			o->diffopts.id_abbrev = 40;
Packit Service 20376f
		}
Packit Service 20376f
		else if (!strcmp(a, "--color"))
Packit Service 20376f
			o->color = 0;
Packit Service 20376f
		else if (!strcmp(a, "--no-color"))
Packit Service 20376f
			o->color = -1;
Packit Service 20376f
		else if (!strcmp(a, "-R"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_REVERSE;
Packit Service 20376f
		else if (!strcmp(a, "-a") || !strcmp(a, "--text"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_FORCE_TEXT;
Packit Service 20376f
		else if (!strcmp(a, "--ignore-space-at-eol"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_IGNORE_WHITESPACE_EOL;
Packit Service 20376f
		else if (!strcmp(a, "-b") || !strcmp(a, "--ignore-space-change"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_IGNORE_WHITESPACE_CHANGE;
Packit Service 20376f
		else if (!strcmp(a, "-w") || !strcmp(a, "--ignore-all-space"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_IGNORE_WHITESPACE;
Packit Service 20376f
		else if (!strcmp(a, "--ignored"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_INCLUDE_IGNORED;
Packit Service 20376f
		else if (!strcmp(a, "--untracked"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_INCLUDE_UNTRACKED;
Packit Service 20376f
		else if (!strcmp(a, "--patience"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_PATIENCE;
Packit Service 20376f
		else if (!strcmp(a, "--minimal"))
Packit Service 20376f
			o->diffopts.flags |= GIT_DIFF_MINIMAL;
Packit Service 20376f
		else if (!strcmp(a, "--stat"))
Packit Service 20376f
			o->output |= OUTPUT_STAT;
Packit Service 20376f
		else if (!strcmp(a, "--numstat"))
Packit Service 20376f
			o->output |= OUTPUT_NUMSTAT;
Packit Service 20376f
		else if (!strcmp(a, "--shortstat"))
Packit Service 20376f
			o->output |= OUTPUT_SHORTSTAT;
Packit Service 20376f
		else if (!strcmp(a, "--summary"))
Packit Service 20376f
			o->output |= OUTPUT_SUMMARY;
Packit Service 20376f
		else if (match_uint16_arg(
Packit Service 20376f
				&o->findopts.rename_threshold, &args, "-M") ||
Packit Service 20376f
			match_uint16_arg(
Packit Service 20376f
				&o->findopts.rename_threshold, &args, "--find-renames"))
Packit Service 20376f
			o->findopts.flags |= GIT_DIFF_FIND_RENAMES;
Packit Service 20376f
		else if (match_uint16_arg(
Packit Service 20376f
				&o->findopts.copy_threshold, &args, "-C") ||
Packit Service 20376f
			match_uint16_arg(
Packit Service 20376f
				&o->findopts.copy_threshold, &args, "--find-copies"))
Packit Service 20376f
			o->findopts.flags |= GIT_DIFF_FIND_COPIES;
Packit Service 20376f
		else if (!strcmp(a, "--find-copies-harder"))
Packit Service 20376f
			o->findopts.flags |= GIT_DIFF_FIND_COPIES_FROM_UNMODIFIED;
Packit Service 20376f
		else if (is_prefixed(a, "-B") || is_prefixed(a, "--break-rewrites"))
Packit Service 20376f
			/* TODO: parse thresholds */
Packit Service 20376f
			o->findopts.flags |= GIT_DIFF_FIND_REWRITES;
Packit Service 20376f
		else if (!match_uint32_arg(
Packit Service 20376f
				&o->diffopts.context_lines, &args, "-U") &&
Packit Service 20376f
			!match_uint32_arg(
Packit Service 20376f
				&o->diffopts.context_lines, &args, "--unified") &&
Packit Service 20376f
			!match_uint32_arg(
Packit Service 20376f
				&o->diffopts.interhunk_lines, &args, "--inter-hunk-context") &&
Packit Service 20376f
			!match_uint16_arg(
Packit Service 20376f
				&o->diffopts.id_abbrev, &args, "--abbrev") &&
Packit Service 20376f
			!match_str_arg(&o->diffopts.old_prefix, &args, "--src-prefix") &&
Packit Service 20376f
			!match_str_arg(&o->diffopts.new_prefix, &args, "--dst-prefix") &&
Packit Service 20376f
			!match_str_arg(&o->dir, &args, "--git-dir"))
Packit Service 20376f
			usage("Unknown command line argument", a);
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/** Display diff output with "--stat", "--numstat", or "--shortstat" */
Packit Service 20376f
static void diff_print_stats(git_diff *diff, struct opts *o)
Packit Service 20376f
{
Packit Service 20376f
	git_diff_stats *stats;
Packit Service 20376f
	git_buf b = GIT_BUF_INIT_CONST(NULL, 0);
Packit Service 20376f
	git_diff_stats_format_t format = 0;
Packit Service 20376f
Packit Service 20376f
	check_lg2(
Packit Service 20376f
		git_diff_get_stats(&stats, diff), "generating stats for diff", NULL);
Packit Service 20376f
Packit Service 20376f
	if (o->output & OUTPUT_STAT)
Packit Service 20376f
		format |= GIT_DIFF_STATS_FULL;
Packit Service 20376f
	if (o->output & OUTPUT_SHORTSTAT)
Packit Service 20376f
		format |= GIT_DIFF_STATS_SHORT;
Packit Service 20376f
	if (o->output & OUTPUT_NUMSTAT)
Packit Service 20376f
		format |= GIT_DIFF_STATS_NUMBER;
Packit Service 20376f
	if (o->output & OUTPUT_SUMMARY)
Packit Service 20376f
		format |= GIT_DIFF_STATS_INCLUDE_SUMMARY;
Packit Service 20376f
Packit Service 20376f
	check_lg2(
Packit Service 20376f
		git_diff_stats_to_buf(&b, stats, format, 80), "formatting stats", NULL);
Packit Service 20376f
Packit Service 20376f
	fputs(b.ptr, stdout);
Packit Service 20376f
Packit Service 20376f
	git_buf_free(&b);
Packit Service 20376f
	git_diff_stats_free(stats);
Packit Service 20376f
}