From: Hiltjo Posthuma Date: Wed, 6 Jan 2016 16:45:02 +0000 (+0100) Subject: use git_reference for tags and branches, sort branches also X-Git-Url: https://git.atheridis.org/?a=commitdiff_plain;h=87360fe2526f7713c4626d04da521579141fcf68;p=forks%2Fstagit.git use git_reference for tags and branches, sort branches also this removes some lines and somewhat simplifies it --- diff --git a/stagit.c b/stagit.c index 73eb723..c35a55b 100644 --- a/stagit.c +++ b/stagit.c @@ -775,199 +775,140 @@ err: } int -writebranches(FILE *fp) +refs_cmp(const void *v1, const void *v2) { - struct commitinfo *ci; - git_branch_iterator *it = NULL; - git_branch_t branch; - git_reference *ref = NULL, *dref = NULL; - const git_oid *id = NULL; - const char *branchname = NULL; - size_t len; - int ret = -1; - - /* log for local branches */ - if (git_branch_iterator_new(&it, repo, GIT_BRANCH_LOCAL)) - return -1; - - fputs("

Branches

\n" - "" - "" - "\n\n", fp); - - while (!git_branch_next(&ref, &branch, it)) { - if (git_branch_name(&branchname, ref)) - continue; - - id = NULL; - switch (git_reference_type(ref)) { - case GIT_REF_SYMBOLIC: - if (git_reference_resolve(&dref, ref)) - goto err; - id = git_reference_target(dref); - break; - case GIT_REF_OID: - id = git_reference_target(ref); - break; - default: - continue; - } - if (!id) - goto err; - if (!(ci = commitinfo_getbyoid(id))) - break; - - relpath = ""; - - fputs("\n", fp); - - relpath = "../"; + git_reference *r1 = (*(git_reference **)v1); + git_reference *r2 = (*(git_reference **)v2); + int t1, t2; - commitinfo_free(ci); - git_reference_free(ref); - git_reference_free(dref); - ref = NULL; - dref = NULL; - } - ret = 0; + t1 = git_reference_is_branch(r1); + t2 = git_reference_is_branch(r2); -err: - fputs("
BranchAgeCommit messageAuthorFiles+-
", fp); - xmlencode(fp, branchname, strlen(branchname)); - fputs("", fp); - if (ci->author) - printtimeshort(fp, &(ci->author->when)); - fputs("", fp); - if (ci->summary) { - if ((len = strlen(ci->summary)) > summarylen) { - xmlencode(fp, ci->summary, summarylen - 1); - fputs("…", fp); - } else { - xmlencode(fp, ci->summary, len); - } - } - fputs("", fp); - if (ci->author) - xmlencode(fp, ci->author->name, strlen(ci->author->name)); - fputs("", fp); - fprintf(fp, "%zu", ci->filecount); - fputs("", fp); - fprintf(fp, "+%zu", ci->addcount); - fputs("", fp); - fprintf(fp, "-%zu", ci->delcount); - fputs("
", fp); - git_reference_free(ref); - git_reference_free(dref); - git_branch_iterator_free(it); + if (t1 != t2) + return t1 - t2; - return ret; + return strcmp(git_reference_shorthand(r1), + git_reference_shorthand(r2)); } int -tagcompare(void *s1, void *s2) -{ - return strcmp(*(char **)s1, *(char **)s2); -} - -int -writetags(FILE *fp) +writerefs(FILE *fp) { struct commitinfo *ci; - git_strarray tagnames; - git_object *obj = NULL; - git_tag *tag = NULL; const git_oid *id = NULL; - size_t i, len; - - /* summary page with branches and tags */ - memset(&tagnames, 0, sizeof(tagnames)); - if (git_tag_list(&tagnames, repo)) + git_object *obj = NULL; + git_reference *dref = NULL, *r, *ref = NULL; + git_reference_iterator *it = NULL; + git_reference **refs = NULL; + size_t count, i, j, len, refcount = 0; + const char *cols[] = { "Branch", "Tag" }; /* first column title */ + const char *titles[] = { "Branches", "Tags" }; + const char *ids[] = { "branches", "tags" }; + const char *name; + + if (git_reference_iterator_new(&it, repo)) return -1; - if (!tagnames.count) { - git_strarray_free(&tagnames); - return 0; - } - - /* sort names */ - qsort(tagnames.strings, tagnames.count, sizeof(char *), - (int (*)(const void *, const void *))&tagcompare); - - fputs("

Tags

\n" - "" - "" - "\n\n", fp); - - for (i = 0; i < tagnames.count; i++) { - if (git_revparse_single(&obj, repo, tagnames.strings[i])) - continue; - id = git_object_id(obj); - /* lookup actual commit (from annotated tag etc) */ - if (!git_tag_lookup(&tag, repo, id)) { - git_object_free(obj); - obj = NULL; - if (git_tag_peel(&obj, tag)) + for (refcount = 0; !git_reference_next(&ref, it); refcount++) { + if (!(refs = reallocarray(refs, refcount + 1, sizeof(git_reference *)))) + err(1, "realloc"); + refs[refcount] = ref; + } + git_reference_iterator_free(it); + + /* sort by type then shorthand name */ + qsort(refs, refcount, sizeof(git_reference *), refs_cmp); + + for (j = 0; j < 2; j++) { + for (i = 0, count = 0; i < refcount; i++) { + if (git_reference_is_branch(refs[i]) && j == 0) + ; + else if (git_reference_is_tag(refs[i]) && j == 1) + ; + else + continue; + + id = NULL; + r = NULL; + switch (git_reference_type(refs[i])) { + case GIT_REF_SYMBOLIC: + if (git_reference_resolve(&dref, refs[i])) + goto err; + r = dref; + break; + case GIT_REF_OID: + r = refs[i]; + break; + default: + continue; + } + if (!(id = git_reference_target(r))) + goto err; + if (git_reference_peel(&obj, r, GIT_OBJ_ANY)) + goto err; + if (!(id = git_object_id(obj))) + goto err; + if (!(ci = commitinfo_getbyoid(id))) break; - git_tag_free(tag); - tag = NULL; - id = git_object_id(obj); - } - - if (!(ci = commitinfo_getbyoid(id))) - break; - relpath = ""; + /* print header if it has an entry (first). */ + if (++count == 1) { + fprintf(fp, "

%s

TagAgeCommit messageAuthorFiles+-
\n" + "" + "" + "\n\n", + titles[j], ids[j], cols[j]); + } - fputs("\n", fp); + + relpath = "../"; + + commitinfo_free(ci); + git_reference_free(dref); + dref = NULL; } - fputs("\n", fp); - - relpath = "../"; - - commitinfo_free(ci); - git_object_free(obj); - obj = NULL; + /* table footer */ + if (count) + fputs("
%sAgeCommit messageAuthorFiles+-
", fp); - xmlencode(fp, tagnames.strings[i], strlen(tagnames.strings[i])); - fputs("", fp); - if (ci->author) - printtimeshort(fp, &(ci->author->when)); - fputs("", fp); - if (ci->summary) { - fprintf(fp, "", relpath, ci->oid); - if ((len = strlen(ci->summary)) > summarylen) { - xmlencode(fp, ci->summary, summarylen - 1); - fputs("…", fp); - } else { - xmlencode(fp, ci->summary, len); + relpath = ""; + name = git_reference_shorthand(r); + + fputs("
", fp); + xmlencode(fp, name, strlen(name)); + fputs("", fp); + if (ci->author) + printtimeshort(fp, &(ci->author->when)); + fputs("", fp); + if (ci->summary) { + fprintf(fp, "", relpath, ci->oid); + if ((len = strlen(ci->summary)) > summarylen) { + xmlencode(fp, ci->summary, summarylen - 1); + fputs("…", fp); + } else { + xmlencode(fp, ci->summary, len); + } + fputs("", fp); } - fputs("", fp); + fputs("", fp); + if (ci->author) + xmlencode(fp, ci->author->name, strlen(ci->author->name)); + fputs("", fp); + fprintf(fp, "%zu", ci->filecount); + fputs("", fp); + fprintf(fp, "+%zu", ci->addcount); + fputs("", fp); + fprintf(fp, "-%zu", ci->delcount); + fputs("
", fp); - if (ci->author) - xmlencode(fp, ci->author->name, strlen(ci->author->name)); - fputs("", fp); - fprintf(fp, "%zu", ci->filecount); - fputs("", fp); - fprintf(fp, "+%zu", ci->addcount); - fputs("", fp); - fprintf(fp, "-%zu", ci->delcount); - fputs("
", fp); } - fputs("", fp); - git_strarray_free(&tagnames); - git_tag_free(tag); - git_object_free(obj); - return 0; -} +err: + git_reference_free(dref); -int -writerefs(FILE *fp) -{ - int ret; + for (i = 0; i < refcount; i++) + git_reference_free(refs[i]); + free(refs); - if ((ret = writebranches(fp))) - return ret; - fputs("
", fp); - return writetags(fp); + return 0; } int