Blame src/notes.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
Packit Service 20376f
#include "notes.h"
Packit Service 20376f
Packit Service 20376f
#include "git2.h"
Packit Service 20376f
#include "refs.h"
Packit Service 20376f
#include "config.h"
Packit Service 20376f
#include "iterator.h"
Packit Service 20376f
#include "signature.h"
Packit Service 20376f
Packit Service 20376f
static int note_error_notfound(void)
Packit Service 20376f
{
Packit Service 20376f
	giterr_set(GITERR_INVALID, "note could not be found");
Packit Service 20376f
	return GIT_ENOTFOUND;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int find_subtree_in_current_level(
Packit Service 20376f
	git_tree **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_tree *parent,
Packit Service 20376f
	const char *annotated_object_sha,
Packit Service 20376f
	int fanout)
Packit Service 20376f
{
Packit Service 20376f
	size_t i;
Packit Service 20376f
	const git_tree_entry *entry;
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
Packit Service 20376f
	if (parent == NULL)
Packit Service 20376f
		return note_error_notfound();
Packit Service 20376f
Packit Service 20376f
	for (i = 0; i < git_tree_entrycount(parent); i++) {
Packit Service 20376f
		entry = git_tree_entry_byindex(parent, i);
Packit Service 20376f
Packit Service 20376f
		if (!git__ishex(git_tree_entry_name(entry)))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		if (S_ISDIR(git_tree_entry_filemode(entry))
Packit Service 20376f
			&& strlen(git_tree_entry_name(entry)) == 2
Packit Service 20376f
			&& !strncmp(git_tree_entry_name(entry), annotated_object_sha + fanout, 2))
Packit Service 20376f
			return git_tree_lookup(out, repo, git_tree_entry_id(entry));
Packit Service 20376f
Packit Service 20376f
		/* Not a DIR, so do we have an already existing blob? */
Packit Service 20376f
		if (!strcmp(git_tree_entry_name(entry), annotated_object_sha + fanout))
Packit Service 20376f
			return GIT_EEXISTS;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return note_error_notfound();
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int find_subtree_r(git_tree **out, git_tree *root,
Packit Service 20376f
			git_repository *repo, const char *target, int *fanout)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_tree *subtree = NULL;
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
Packit Service 20376f
	error = find_subtree_in_current_level(&subtree, repo, root, target, *fanout);
Packit Service 20376f
	if (error == GIT_EEXISTS)
Packit Service 20376f
		return git_tree_lookup(out, repo, git_tree_id(root));
Packit Service 20376f
Packit Service 20376f
	if (error < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	*fanout += 2;
Packit Service 20376f
	error = find_subtree_r(out, subtree, repo, target, fanout);
Packit Service 20376f
	git_tree_free(subtree);
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int find_blob(git_oid *blob, git_tree *tree, const char *target)
Packit Service 20376f
{
Packit Service 20376f
	size_t i;
Packit Service 20376f
	const git_tree_entry *entry;
Packit Service 20376f
Packit Service 20376f
	for (i=0; i
Packit Service 20376f
		entry = git_tree_entry_byindex(tree, i);
Packit Service 20376f
Packit Service 20376f
		if (!strcmp(git_tree_entry_name(entry), target)) {
Packit Service 20376f
			/* found matching note object - return */
Packit Service 20376f
Packit Service 20376f
			git_oid_cpy(blob, git_tree_entry_id(entry));
Packit Service 20376f
			return 0;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return note_error_notfound();
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_write(
Packit Service 20376f
	git_tree **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_tree *source_tree,
Packit Service 20376f
	const git_oid *object_oid,
Packit Service 20376f
	const char *treeentry_name,
Packit Service 20376f
	unsigned int attributes)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_treebuilder *tb = NULL;
Packit Service 20376f
	const git_tree_entry *entry;
Packit Service 20376f
	git_oid tree_oid;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_treebuilder_new(&tb, repo, source_tree)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	if (object_oid) {
Packit Service 20376f
		if ((error = git_treebuilder_insert(
Packit Service 20376f
				&entry, tb, treeentry_name, object_oid, attributes)) < 0)
Packit Service 20376f
			goto cleanup;
Packit Service 20376f
	} else {
Packit Service 20376f
		if ((error = git_treebuilder_remove(tb, treeentry_name)) < 0)
Packit Service 20376f
			goto cleanup;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if ((error = git_treebuilder_write(&tree_oid, tb)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	error = git_tree_lookup(out, repo, &tree_oid);
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git_treebuilder_free(tb);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int manipulate_note_in_tree_r(
Packit Service 20376f
	git_tree **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_tree *parent,
Packit Service 20376f
	git_oid *note_oid,
Packit Service 20376f
	const char *annotated_object_sha,
Packit Service 20376f
	int fanout,
Packit Service 20376f
	int (*note_exists_cb)(
Packit Service 20376f
		git_tree **out,
Packit Service 20376f
		git_repository *repo,
Packit Service 20376f
		git_tree *parent,
Packit Service 20376f
		git_oid *note_oid,
Packit Service 20376f
		const char *annotated_object_sha,
Packit Service 20376f
		int fanout,
Packit Service 20376f
		int current_error),
Packit Service 20376f
	int (*note_notfound_cb)(
Packit Service 20376f
		git_tree **out,
Packit Service 20376f
		git_repository *repo,
Packit Service 20376f
		git_tree *parent,
Packit Service 20376f
		git_oid *note_oid,
Packit Service 20376f
		const char *annotated_object_sha,
Packit Service 20376f
		int fanout,
Packit Service 20376f
		int current_error))
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_tree *subtree = NULL, *new = NULL;
Packit Service 20376f
	char subtree_name[3];
Packit Service 20376f
Packit Service 20376f
	error = find_subtree_in_current_level(
Packit Service 20376f
		&subtree, repo, parent, annotated_object_sha, fanout);
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_EEXISTS) {
Packit Service 20376f
		error = note_exists_cb(
Packit Service 20376f
			out, repo, parent, note_oid, annotated_object_sha, fanout, error);
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_ENOTFOUND) {
Packit Service 20376f
		error = note_notfound_cb(
Packit Service 20376f
			out, repo, parent, note_oid, annotated_object_sha, fanout, error);
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (error < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	/* An existing fanout has been found, let's dig deeper */
Packit Service 20376f
	error = manipulate_note_in_tree_r(
Packit Service 20376f
		&new, repo, subtree, note_oid, annotated_object_sha,
Packit Service 20376f
		fanout + 2, note_exists_cb, note_notfound_cb);
Packit Service 20376f
Packit Service 20376f
	if (error < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	strncpy(subtree_name, annotated_object_sha + fanout, 2);
Packit Service 20376f
	subtree_name[2] = '\0';
Packit Service 20376f
Packit Service 20376f
	error = tree_write(out, repo, parent, git_tree_id(new),
Packit Service 20376f
			   subtree_name, GIT_FILEMODE_TREE);
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git_tree_free(new);
Packit Service 20376f
	git_tree_free(subtree);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int remove_note_in_tree_eexists_cb(
Packit Service 20376f
	git_tree **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_tree *parent,
Packit Service 20376f
	git_oid *note_oid,
Packit Service 20376f
	const char *annotated_object_sha,
Packit Service 20376f
	int fanout,
Packit Service 20376f
	int current_error)
Packit Service 20376f
{
Packit Service 20376f
	GIT_UNUSED(note_oid);
Packit Service 20376f
	GIT_UNUSED(current_error);
Packit Service 20376f
Packit Service 20376f
	return tree_write(out, repo, parent, NULL, annotated_object_sha + fanout, 0);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int remove_note_in_tree_enotfound_cb(
Packit Service 20376f
	git_tree **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_tree *parent,
Packit Service 20376f
	git_oid *note_oid,
Packit Service 20376f
	const char *annotated_object_sha,
Packit Service 20376f
	int fanout,
Packit Service 20376f
	int current_error)
Packit Service 20376f
{
Packit Service 20376f
	GIT_UNUSED(out);
Packit Service 20376f
	GIT_UNUSED(repo);
Packit Service 20376f
	GIT_UNUSED(parent);
Packit Service 20376f
	GIT_UNUSED(note_oid);
Packit Service 20376f
	GIT_UNUSED(fanout);
Packit Service 20376f
Packit Service 20376f
	giterr_set(GITERR_REPOSITORY, "object '%s' has no note", annotated_object_sha);
Packit Service 20376f
	return current_error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int insert_note_in_tree_eexists_cb(git_tree **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_tree *parent,
Packit Service 20376f
	git_oid *note_oid,
Packit Service 20376f
	const char *annotated_object_sha,
Packit Service 20376f
	int fanout,
Packit Service 20376f
	int current_error)
Packit Service 20376f
{
Packit Service 20376f
	GIT_UNUSED(out);
Packit Service 20376f
	GIT_UNUSED(repo);
Packit Service 20376f
	GIT_UNUSED(parent);
Packit Service 20376f
	GIT_UNUSED(note_oid);
Packit Service 20376f
	GIT_UNUSED(fanout);
Packit Service 20376f
Packit Service 20376f
	giterr_set(GITERR_REPOSITORY, "note for '%s' exists already", annotated_object_sha);
Packit Service 20376f
	return current_error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int insert_note_in_tree_enotfound_cb(git_tree **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_tree *parent,
Packit Service 20376f
	git_oid *note_oid,
Packit Service 20376f
	const char *annotated_object_sha,
Packit Service 20376f
	int fanout,
Packit Service 20376f
	int current_error)
Packit Service 20376f
{
Packit Service 20376f
	GIT_UNUSED(current_error);
Packit Service 20376f
Packit Service 20376f
	/* No existing fanout at this level, insert in place */
Packit Service 20376f
	return tree_write(
Packit Service 20376f
		out,
Packit Service 20376f
		repo,
Packit Service 20376f
		parent,
Packit Service 20376f
		note_oid,
Packit Service 20376f
		annotated_object_sha + fanout,
Packit Service 20376f
		GIT_FILEMODE_BLOB);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int note_write(git_oid *out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	const git_signature *author,
Packit Service 20376f
	const git_signature *committer,
Packit Service 20376f
	const char *notes_ref,
Packit Service 20376f
	const char *note,
Packit Service 20376f
	git_tree *commit_tree,
Packit Service 20376f
	const char *target,
Packit Service 20376f
	git_commit **parents,
Packit Service 20376f
	int allow_note_overwrite)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_oid oid;
Packit Service 20376f
	git_tree *tree = NULL;
Packit Service 20376f
Packit Service 20376f
	// TODO: should we apply filters?
Packit Service 20376f
	/* create note object */
Packit Service 20376f
	if ((error = git_blob_create_frombuffer(&oid, repo, note, strlen(note))) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	if ((error = manipulate_note_in_tree_r(
Packit Service 20376f
		&tree, repo, commit_tree, &oid, target, 0,
Packit Service 20376f
		allow_note_overwrite ? insert_note_in_tree_enotfound_cb : insert_note_in_tree_eexists_cb,
Packit Service 20376f
		insert_note_in_tree_enotfound_cb)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	if (out)
Packit Service 20376f
		git_oid_cpy(out, &oid;;
Packit Service 20376f
Packit Service 20376f
	error = git_commit_create(&oid, repo, notes_ref, author, committer,
Packit Service 20376f
				  NULL, GIT_NOTES_DEFAULT_MSG_ADD,
Packit Service 20376f
				  tree, *parents == NULL ? 0 : 1, (const git_commit **) parents);
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git_tree_free(tree);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int note_new(
Packit Service 20376f
	git_note **out,
Packit Service 20376f
	git_oid *note_oid,
Packit Service 20376f
	git_commit *commit,
Packit Service 20376f
	git_blob *blob)
Packit Service 20376f
{
Packit Service 20376f
	git_note *note = NULL;
Packit Service 20376f
Packit Service 20376f
	note = git__malloc(sizeof(git_note));
Packit Service 20376f
	GITERR_CHECK_ALLOC(note);
Packit Service 20376f
Packit Service 20376f
	git_oid_cpy(&note->id, note_oid);
Packit Service 20376f
Packit Service 20376f
	if (git_signature_dup(&note->author, git_commit_author(commit)) < 0 ||
Packit Service 20376f
		git_signature_dup(&note->committer, git_commit_committer(commit)) < 0)
Packit Service 20376f
		return -1;
Packit Service 20376f
Packit Service 20376f
	note->message = git__strndup(git_blob_rawcontent(blob), git_blob_rawsize(blob));
Packit Service 20376f
	GITERR_CHECK_ALLOC(note->message);
Packit Service 20376f
Packit Service 20376f
	*out = note;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int note_lookup(
Packit Service 20376f
	git_note **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_commit *commit,
Packit Service 20376f
	git_tree *tree,
Packit Service 20376f
	const char *target)
Packit Service 20376f
{
Packit Service 20376f
	int error, fanout = 0;
Packit Service 20376f
	git_oid oid;
Packit Service 20376f
	git_blob *blob = NULL;
Packit Service 20376f
	git_note *note = NULL;
Packit Service 20376f
	git_tree *subtree = NULL;
Packit Service 20376f
Packit Service 20376f
	if ((error = find_subtree_r(&subtree, tree, repo, target, &fanout)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	if ((error = find_blob(&oid, subtree, target + fanout)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_blob_lookup(&blob, repo, &oid)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	if ((error = note_new(&note, &oid, commit, blob)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	*out = note;
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git_tree_free(subtree);
Packit Service 20376f
	git_blob_free(blob);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int note_remove(git_repository *repo,
Packit Service 20376f
		const git_signature *author, const git_signature *committer,
Packit Service 20376f
		const char *notes_ref, git_tree *tree,
Packit Service 20376f
		const char *target, git_commit **parents)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_tree *tree_after_removal = NULL;
Packit Service 20376f
	git_oid oid;
Packit Service 20376f
Packit Service 20376f
	if ((error = manipulate_note_in_tree_r(
Packit Service 20376f
		&tree_after_removal, repo, tree, NULL, target, 0,
Packit Service 20376f
		remove_note_in_tree_eexists_cb, remove_note_in_tree_enotfound_cb)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	error = git_commit_create(&oid, repo, notes_ref, author, committer,
Packit Service 20376f
	  NULL, GIT_NOTES_DEFAULT_MSG_RM,
Packit Service 20376f
	  tree_after_removal,
Packit Service 20376f
	  *parents == NULL ? 0 : 1,
Packit Service 20376f
	  (const git_commit **) parents);
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git_tree_free(tree_after_removal);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int note_get_default_ref(char **out, git_repository *repo)
Packit Service 20376f
{
Packit Service 20376f
	git_config *cfg;
Packit Service 20376f
	int ret = git_repository_config__weakptr(&cfg, repo);
Packit Service 20376f
Packit Service 20376f
	*out = (ret != 0) ? NULL : git_config__get_string_force(
Packit Service 20376f
		cfg, "core.notesref", GIT_NOTES_DEFAULT_REF);
Packit Service 20376f
Packit Service 20376f
	return ret;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int normalize_namespace(char **out, git_repository *repo, const char *notes_ref)
Packit Service 20376f
{
Packit Service 20376f
	if (notes_ref) {
Packit Service 20376f
		*out = git__strdup(notes_ref);
Packit Service 20376f
		GITERR_CHECK_ALLOC(*out);
Packit Service 20376f
		return 0;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return note_get_default_ref(out, repo);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int retrieve_note_tree_and_commit(
Packit Service 20376f
	git_tree **tree_out,
Packit Service 20376f
	git_commit **commit_out,
Packit Service 20376f
	char **notes_ref_out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	const char *notes_ref)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_oid oid;
Packit Service 20376f
Packit Service 20376f
	if ((error = normalize_namespace(notes_ref_out, repo, notes_ref)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_reference_name_to_id(&oid, repo, *notes_ref_out)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if (git_commit_lookup(commit_out, repo, &oid) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_commit_tree(tree_out, *commit_out)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_note_read(git_note **out, git_repository *repo,
Packit Service 20376f
		  const char *notes_ref_in, const git_oid *oid)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	char *target = NULL, *notes_ref = NULL;
Packit Service 20376f
	git_tree *tree = NULL;
Packit Service 20376f
	git_commit *commit = NULL;
Packit Service 20376f
Packit Service 20376f
	target = git_oid_allocfmt(oid);
Packit Service 20376f
	GITERR_CHECK_ALLOC(target);
Packit Service 20376f
Packit Service 20376f
	if (!(error = retrieve_note_tree_and_commit(
Packit Service 20376f
		      &tree, &commit, &notes_ref, repo, notes_ref_in)))
Packit Service 20376f
		error = note_lookup(out, repo, commit, tree, target);
Packit Service 20376f
Packit Service 20376f
	git__free(notes_ref);
Packit Service 20376f
	git__free(target);
Packit Service 20376f
	git_tree_free(tree);
Packit Service 20376f
	git_commit_free(commit);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_note_create(
Packit Service 20376f
	git_oid *out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	const char *notes_ref_in,
Packit Service 20376f
	const git_signature *author,
Packit Service 20376f
	const git_signature *committer,
Packit Service 20376f
	const git_oid *oid,
Packit Service 20376f
	const char *note,
Packit Service 20376f
	int allow_note_overwrite)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	char *target = NULL, *notes_ref = NULL;
Packit Service 20376f
	git_commit *commit = NULL;
Packit Service 20376f
	git_tree *tree = NULL;
Packit Service 20376f
Packit Service 20376f
	target = git_oid_allocfmt(oid);
Packit Service 20376f
	GITERR_CHECK_ALLOC(target);
Packit Service 20376f
Packit Service 20376f
	error = retrieve_note_tree_and_commit(&tree, &commit, &notes_ref, repo, notes_ref_in);
Packit Service 20376f
Packit Service 20376f
	if (error < 0 && error != GIT_ENOTFOUND)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	error = note_write(out, repo, author, committer, notes_ref,
Packit Service 20376f
			note, tree, target, &commit, allow_note_overwrite);
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git__free(notes_ref);
Packit Service 20376f
	git__free(target);
Packit Service 20376f
	git_commit_free(commit);
Packit Service 20376f
	git_tree_free(tree);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_note_remove(git_repository *repo, const char *notes_ref_in,
Packit Service 20376f
		const git_signature *author, const git_signature *committer,
Packit Service 20376f
		const git_oid *oid)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	char *target = NULL, *notes_ref;
Packit Service 20376f
	git_commit *commit = NULL;
Packit Service 20376f
	git_tree *tree = NULL;
Packit Service 20376f
Packit Service 20376f
	target = git_oid_allocfmt(oid);
Packit Service 20376f
	GITERR_CHECK_ALLOC(target);
Packit Service 20376f
Packit Service 20376f
	if (!(error = retrieve_note_tree_and_commit(
Packit Service 20376f
		      &tree, &commit, &notes_ref, repo, notes_ref_in)))
Packit Service 20376f
		error = note_remove(
Packit Service 20376f
			repo, author, committer, notes_ref, tree, target, &commit);
Packit Service 20376f
Packit Service 20376f
	git__free(notes_ref);
Packit Service 20376f
	git__free(target);
Packit Service 20376f
	git_commit_free(commit);
Packit Service 20376f
	git_tree_free(tree);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_note_default_ref(git_buf *out, git_repository *repo)
Packit Service 20376f
{
Packit Service 20376f
	char *default_ref;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	assert(out && repo);
Packit Service 20376f
Packit Service 20376f
	git_buf_sanitize(out);
Packit Service 20376f
Packit Service 20376f
	if ((error = note_get_default_ref(&default_ref, repo)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	git_buf_attach(out, default_ref, strlen(default_ref));
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
const git_signature *git_note_committer(const git_note *note)
Packit Service 20376f
{
Packit Service 20376f
	assert(note);
Packit Service 20376f
	return note->committer;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
const git_signature *git_note_author(const git_note *note)
Packit Service 20376f
{
Packit Service 20376f
	assert(note);
Packit Service 20376f
	return note->author;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
const char * git_note_message(const git_note *note)
Packit Service 20376f
{
Packit Service 20376f
	assert(note);
Packit Service 20376f
	return note->message;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
const git_oid * git_note_id(const git_note *note)
Packit Service 20376f
{
Packit Service 20376f
	assert(note);
Packit Service 20376f
	return &note->id;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void git_note_free(git_note *note)
Packit Service 20376f
{
Packit Service 20376f
	if (note == NULL)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	git_signature_free(note->committer);
Packit Service 20376f
	git_signature_free(note->author);
Packit Service 20376f
	git__free(note->message);
Packit Service 20376f
	git__free(note);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int process_entry_path(
Packit Service 20376f
	const char* entry_path,
Packit Service 20376f
	git_oid *annotated_object_id)
Packit Service 20376f
{
Packit Service 20376f
	int error = 0;
Packit Service 20376f
	size_t i = 0, j = 0, len;
Packit Service 20376f
	git_buf buf = GIT_BUF_INIT;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_buf_puts(&buf, entry_path)) < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	len = git_buf_len(&buf;;
Packit Service 20376f
Packit Service 20376f
	while (i < len) {
Packit Service 20376f
		if (buf.ptr[i] == '/') {
Packit Service 20376f
			i++;
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (git__fromhex(buf.ptr[i]) < 0) {
Packit Service 20376f
			/* This is not a note entry */
Packit Service 20376f
			goto cleanup;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (i != j)
Packit Service 20376f
			buf.ptr[j] = buf.ptr[i];
Packit Service 20376f
Packit Service 20376f
		i++;
Packit Service 20376f
		j++;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	buf.ptr[j] = '\0';
Packit Service 20376f
	buf.size = j;
Packit Service 20376f
Packit Service 20376f
	if (j != GIT_OID_HEXSZ) {
Packit Service 20376f
		/* This is not a note entry */
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	error = git_oid_fromstr(annotated_object_id, buf.ptr);
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git_buf_free(&buf;;
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_note_foreach(
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	const char *notes_ref,
Packit Service 20376f
	git_note_foreach_cb note_cb,
Packit Service 20376f
	void *payload)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_note_iterator *iter = NULL;
Packit Service 20376f
	git_oid note_id, annotated_id;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_note_iterator_new(&iter, repo, notes_ref)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	while (!(error = git_note_next(&note_id, &annotated_id, iter))) {
Packit Service 20376f
		if ((error = note_cb(&note_id, &annotated_id, payload)) != 0) {
Packit Service 20376f
			giterr_set_after_callback(error);
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_ITEROVER)
Packit Service 20376f
		error = 0;
Packit Service 20376f
Packit Service 20376f
	git_note_iterator_free(iter);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
void git_note_iterator_free(git_note_iterator *it)
Packit Service 20376f
{
Packit Service 20376f
	if (it == NULL)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	git_iterator_free(it);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
int git_note_iterator_new(
Packit Service 20376f
	git_note_iterator **it,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	const char *notes_ref_in)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	git_commit *commit = NULL;
Packit Service 20376f
	git_tree *tree = NULL;
Packit Service 20376f
	char *notes_ref;
Packit Service 20376f
Packit Service 20376f
	error = retrieve_note_tree_and_commit(&tree, &commit, &notes_ref, repo, notes_ref_in);
Packit Service 20376f
	if (error < 0)
Packit Service 20376f
		goto cleanup;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_iterator_for_tree(it, tree, NULL)) < 0)
Packit Service 20376f
		git_iterator_free(*it);
Packit Service 20376f
Packit Service 20376f
cleanup:
Packit Service 20376f
	git__free(notes_ref);
Packit Service 20376f
	git_tree_free(tree);
Packit Service 20376f
	git_commit_free(commit);
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_note_next(
Packit Service 20376f
	git_oid* note_id,
Packit Service 20376f
	git_oid* annotated_id,
Packit Service 20376f
	git_note_iterator *it)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
	const git_index_entry *item;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_iterator_current(&item, it)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	git_oid_cpy(note_id, &item->id);
Packit Service 20376f
Packit Service 20376f
	if (!(error = process_entry_path(item->path, annotated_id)))
Packit Service 20376f
		git_iterator_advance(NULL, it);
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}