Blame src/diff.c

Packit Service 20376f
/*
Packit Service 20376f
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit Service 20376f
 *
Packit Service 20376f
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit Service 20376f
 * a Linking Exception. For full terms see the included COPYING file.
Packit Service 20376f
 */
Packit Service 20376f
#include "git2/version.h"
Packit Service 20376f
#include "common.h"
Packit Service 20376f
#include "diff.h"
Packit Service 20376f
#include "diff_generate.h"
Packit Service 20376f
#include "patch.h"
Packit Service 20376f
#include "commit.h"
Packit Service 20376f
#include "index.h"
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(const char *) diff_delta__path(const git_diff_delta *delta)
Packit Service 20376f
{
Packit Service 20376f
	const char *str = delta->old_file.path;
Packit Service 20376f
Packit Service 20376f
	if (!str ||
Packit Service 20376f
		delta->status == GIT_DELTA_ADDED ||
Packit Service 20376f
		delta->status == GIT_DELTA_RENAMED ||
Packit Service 20376f
		delta->status == GIT_DELTA_COPIED)
Packit Service 20376f
		str = delta->new_file.path;
Packit Service 20376f
Packit Service 20376f
	return str;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
const char *git_diff_delta__path(const git_diff_delta *delta)
Packit Service 20376f
{
Packit Service 20376f
	return diff_delta__path(delta);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_delta__cmp(const void *a, const void *b)
Packit Service 20376f
{
Packit Service 20376f
	const git_diff_delta *da = a, *db = b;
Packit Service 20376f
	int val = strcmp(diff_delta__path(da), diff_delta__path(db));
Packit Service 20376f
	return val ? val : ((int)da->status - (int)db->status);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_delta__casecmp(const void *a, const void *b)
Packit Service 20376f
{
Packit Service 20376f
	const git_diff_delta *da = a, *db = b;
Packit Service 20376f
	int val = strcasecmp(diff_delta__path(da), diff_delta__path(db));
Packit Service 20376f
	return val ? val : ((int)da->status - (int)db->status);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff__entry_cmp(const void *a, const void *b)
Packit Service 20376f
{
Packit Service 20376f
	const git_index_entry *entry_a = a;
Packit Service 20376f
	const git_index_entry *entry_b = b;
Packit Service 20376f
Packit Service 20376f
	return strcmp(entry_a->path, entry_b->path);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff__entry_icmp(const void *a, const void *b)
Packit Service 20376f
{
Packit Service 20376f
	const git_index_entry *entry_a = a;
Packit Service 20376f
	const git_index_entry *entry_b = b;
Packit Service 20376f
Packit Service 20376f
	return strcasecmp(entry_a->path, entry_b->path);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void git_diff_free(git_diff *diff)
Packit Service 20376f
{
Packit Service 20376f
	if (!diff)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	GIT_REFCOUNT_DEC(diff, diff->free_fn);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void git_diff_addref(git_diff *diff)
Packit Service 20376f
{
Packit Service 20376f
	GIT_REFCOUNT_INC(diff);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
size_t git_diff_num_deltas(const git_diff *diff)
Packit Service 20376f
{
Packit Service 20376f
	assert(diff);
Packit Service 20376f
	return diff->deltas.length;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
size_t git_diff_num_deltas_of_type(const git_diff *diff, git_delta_t type)
Packit Service 20376f
{
Packit Service 20376f
	size_t i, count = 0;
Packit Service 20376f
	const git_diff_delta *delta;
Packit Service 20376f
Packit Service 20376f
	assert(diff);
Packit Service 20376f
Packit Service 20376f
	git_vector_foreach(&diff->deltas, i, delta) {
Packit Service 20376f
		count += (delta->status == type);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return count;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
const git_diff_delta *git_diff_get_delta(const git_diff *diff, size_t idx)
Packit Service 20376f
{
Packit Service 20376f
	assert(diff);
Packit Service 20376f
	return git_vector_get(&diff->deltas, idx);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_is_sorted_icase(const git_diff *diff)
Packit Service 20376f
{
Packit Service 20376f
	return (diff->opts.flags & GIT_DIFF_IGNORE_CASE) != 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_get_perfdata(git_diff_perfdata *out, const git_diff *diff)
Packit Service 20376f
{
Packit Service 20376f
	assert(out);
Packit Service 20376f
	GITERR_CHECK_VERSION(out, GIT_DIFF_PERFDATA_VERSION, "git_diff_perfdata");
Packit Service 20376f
	out->stat_calls = diff->perf.stat_calls;
Packit Service 20376f
	out->oid_calculations = diff->perf.oid_calculations;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_foreach(
Packit Service 20376f
	git_diff *diff,
Packit Service 20376f
	git_diff_file_cb file_cb,
Packit Service 20376f
	git_diff_binary_cb binary_cb,
Packit Service 20376f
	git_diff_hunk_cb hunk_cb,
Packit Service 20376f
	git_diff_line_cb data_cb,
Packit Service 20376f
	void *payload)
Packit Service 20376f
{
Packit Service 20376f
	int error = 0;
Packit Service 20376f
	git_diff_delta *delta;
Packit Service 20376f
	size_t idx;
Packit Service 20376f
Packit Service 20376f
	assert(diff);
Packit Service 20376f
Packit Service 20376f
	git_vector_foreach(&diff->deltas, idx, delta) {
Packit Service 20376f
		git_patch *patch;
Packit Service 20376f
Packit Service 20376f
		/* check flags against patch status */
Packit Service 20376f
		if (git_diff_delta__should_skip(&diff->opts, delta))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		if ((error = git_patch_from_diff(&patch, diff, idx)) != 0)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		error = git_patch__invoke_callbacks(patch, file_cb, binary_cb,
Packit Service 20376f
						    hunk_cb, data_cb, payload);
Packit Service 20376f
		git_patch_free(patch);
Packit Service 20376f
Packit Service 20376f
		if (error)
Packit Service 20376f
			break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_format_email__append_header_tobuf(
Packit Service 20376f
	git_buf *out,
Packit Service 20376f
	const git_oid *id,
Packit Service 20376f
	const git_signature *author,
Packit Service 20376f
	const char *summary,
Packit Service 20376f
	const char *body,
Packit Service 20376f
	size_t patch_no,
Packit Service 20376f
	size_t total_patches,
Packit Service 20376f
	bool exclude_patchno_marker)
Packit Service 20376f
{
Packit Service 20376f
	char idstr[GIT_OID_HEXSZ + 1];
Packit Service 20376f
	char date_str[GIT_DATE_RFC2822_SZ];
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	git_oid_fmt(idstr, id);
Packit Service 20376f
	idstr[GIT_OID_HEXSZ] = '\0';
Packit Service 20376f
Packit Service 20376f
	if ((error = git__date_rfc2822_fmt(date_str, sizeof(date_str),
Packit Service 20376f
		&author->when)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	error = git_buf_printf(out,
Packit Service 20376f
				"From %s Mon Sep 17 00:00:00 2001\n" \
Packit Service 20376f
				"From: %s <%s>\n" \
Packit Service 20376f
				"Date: %s\n" \
Packit Service 20376f
				"Subject: ",
Packit Service 20376f
				idstr,
Packit Service 20376f
				author->name, author->email,
Packit Service 20376f
				date_str);
Packit Service 20376f
Packit Service 20376f
	if (error < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if (!exclude_patchno_marker) {
Packit Service 20376f
		if (total_patches == 1) {
Packit Service 20376f
			error = git_buf_puts(out, "[PATCH] ");
Packit Service 20376f
		} else {
Packit Service 20376f
			error = git_buf_printf(out, "[PATCH %"PRIuZ"/%"PRIuZ"] ",
Packit Service 20376f
				patch_no, total_patches);
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (error < 0)
Packit Service 20376f
			return error;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	error = git_buf_printf(out, "%s\n\n", summary);
Packit Service 20376f
Packit Service 20376f
	if (body) {
Packit Service 20376f
		git_buf_puts(out, body);
Packit Service 20376f
Packit Service 20376f
		if (out->ptr[out->size - 1] != '\n')
Packit Service 20376f
			git_buf_putc(out, '\n');
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_format_email__append_patches_tobuf(
Packit Service 20376f
	git_buf *out,
Packit Service 20376f
	git_diff *diff)
Packit Service 20376f
{
Packit Service 20376f
	size_t i, deltas;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	deltas = git_diff_num_deltas(diff);
Packit Service 20376f
Packit Service 20376f
	for (i = 0; i < deltas; ++i) {
Packit Service 20376f
		git_patch *patch = NULL;
Packit Service 20376f
Packit Service 20376f
		if ((error = git_patch_from_diff(&patch, diff, i)) >= 0)
Packit Service 20376f
			error = git_patch_to_buf(out, patch);
Packit Service 20376f
Packit Service 20376f
		git_patch_free(patch);
Packit Service 20376f
Packit Service 20376f
		if (error < 0)
Packit Service 20376f
			break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_format_email(
Packit Service 20376f
	git_buf *out,
Packit Service 20376f
	git_diff *diff,
Packit Service 20376f
	const git_diff_format_email_options *opts)
Packit Service 20376f
{
Packit Service 20376f
	git_diff_stats *stats = NULL;
Packit Service 20376f
	char *summary = NULL, *loc = NULL;
Packit Service 20376f
	bool ignore_marker;
Packit Service 20376f
	unsigned int format_flags = 0;
Packit Service 20376f
	size_t allocsize;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	assert(out && diff && opts);
Packit Service 20376f
	assert(opts->summary && opts->id && opts->author);
Packit Service 20376f
Packit Service 20376f
	GITERR_CHECK_VERSION(opts,
Packit Service 20376f
		GIT_DIFF_FORMAT_EMAIL_OPTIONS_VERSION,
Packit Service 20376f
		"git_format_email_options");
Packit Service 20376f
Packit Service 20376f
	ignore_marker = (opts->flags &
Packit Service 20376f
		GIT_DIFF_FORMAT_EMAIL_EXCLUDE_SUBJECT_PATCH_MARKER) != 0;
Packit Service 20376f
Packit Service 20376f
	if (!ignore_marker) {
Packit Service 20376f
		if (opts->patch_no > opts->total_patches) {
Packit Service 20376f
			giterr_set(GITERR_INVALID,
Packit Service 20376f
				"patch %"PRIuZ" out of range. max %"PRIuZ,
Packit Service 20376f
				opts->patch_no, opts->total_patches);
Packit Service 20376f
			return -1;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (opts->patch_no == 0) {
Packit Service 20376f
			giterr_set(GITERR_INVALID,
Packit Service 20376f
				"invalid patch no %"PRIuZ". should be >0", opts->patch_no);
Packit Service 20376f
			return -1;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* the summary we receive may not be clean.
Packit Service 20376f
	 * it could potentially contain new line characters
Packit Service 20376f
	 * or not be set, sanitize, */
Packit Service 20376f
	if ((loc = strpbrk(opts->summary, "\r\n")) != NULL) {
Packit Service 20376f
		size_t offset = 0;
Packit Service 20376f
Packit Service 20376f
		if ((offset = (loc - opts->summary)) == 0) {
Packit Service 20376f
			giterr_set(GITERR_INVALID, "summary is empty");
Packit Service 20376f
			error = -1;
Packit Service 20376f
			goto on_error;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		GITERR_CHECK_ALLOC_ADD(&allocsize, offset, 1);
Packit Service 20376f
		summary = git__calloc(allocsize, sizeof(char));
Packit Service 20376f
		GITERR_CHECK_ALLOC(summary);
Packit Service 20376f
Packit Service 20376f
		strncpy(summary, opts->summary, offset);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	error = git_diff_format_email__append_header_tobuf(out,
Packit Service 20376f
		opts->id, opts->author, summary == NULL ? opts->summary : summary,
Packit Service 20376f
		opts->body, opts->patch_no, opts->total_patches, ignore_marker);
Packit Service 20376f
Packit Service 20376f
	if (error < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	format_flags = GIT_DIFF_STATS_FULL | GIT_DIFF_STATS_INCLUDE_SUMMARY;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_buf_puts(out, "---\n")) < 0 ||
Packit Service 20376f
		(error = git_diff_get_stats(&stats, diff)) < 0 ||
Packit Service 20376f
		(error = git_diff_stats_to_buf(out, stats, format_flags, 0)) < 0 ||
Packit Service 20376f
		(error = git_buf_putc(out, '\n')) < 0 ||
Packit Service 20376f
		(error = git_diff_format_email__append_patches_tobuf(out, diff)) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
Packit Service 20376f
	error = git_buf_puts(out, "--\nlibgit2 " LIBGIT2_VERSION "\n\n");
Packit Service 20376f
Packit Service 20376f
on_error:
Packit Service 20376f
	git__free(summary);
Packit Service 20376f
	git_diff_stats_free(stats);
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_commit_as_email(
Packit Service 20376f
	git_buf *out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_commit *commit,
Packit Service 20376f
	size_t patch_no,
Packit Service 20376f
	size_t total_patches,
Packit Service 20376f
	git_diff_format_email_flags_t flags,
Packit Service 20376f
	const git_diff_options *diff_opts)
Packit Service 20376f
{
Packit Service 20376f
	git_diff *diff = NULL;
Packit Service 20376f
	git_diff_format_email_options opts =
Packit Service 20376f
		GIT_DIFF_FORMAT_EMAIL_OPTIONS_INIT;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	assert (out && repo && commit);
Packit Service 20376f
Packit Service 20376f
	opts.flags = flags;
Packit Service 20376f
	opts.patch_no = patch_no;
Packit Service 20376f
	opts.total_patches = total_patches;
Packit Service 20376f
	opts.id = git_commit_id(commit);
Packit Service 20376f
	opts.summary = git_commit_summary(commit);
Packit Service 20376f
	opts.body = git_commit_body(commit);
Packit Service 20376f
	opts.author = git_commit_author(commit);
Packit Service 20376f
Packit Service 20376f
	if ((error = git_diff__commit(&diff, repo, commit, diff_opts)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	error = git_diff_format_email(out, diff, &opts);
Packit Service 20376f
Packit Service 20376f
	git_diff_free(diff);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_init_options(git_diff_options *opts, unsigned int version)
Packit Service 20376f
{
Packit Service 20376f
	GIT_INIT_STRUCTURE_FROM_TEMPLATE(
Packit Service 20376f
		opts, version, git_diff_options, GIT_DIFF_OPTIONS_INIT);
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_find_init_options(
Packit Service 20376f
	git_diff_find_options *opts, unsigned int version)
Packit Service 20376f
{
Packit Service 20376f
	GIT_INIT_STRUCTURE_FROM_TEMPLATE(
Packit Service 20376f
		opts, version, git_diff_find_options, GIT_DIFF_FIND_OPTIONS_INIT);
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_diff_format_email_init_options(
Packit Service 20376f
	git_diff_format_email_options *opts, unsigned int version)
Packit Service 20376f
{
Packit Service 20376f
	GIT_INIT_STRUCTURE_FROM_TEMPLATE(
Packit Service 20376f
		opts, version, git_diff_format_email_options,
Packit Service 20376f
		GIT_DIFF_FORMAT_EMAIL_OPTIONS_INIT);
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f