Commit f597ea89 by Ben Straub

Implemented partial caret syntax for rev-parse.

Supported forms:
- "^n"
- "^0"
- "^"

Still missing: all of the "^{…}" variants.
parent 023c6f69
...@@ -14,6 +14,8 @@ ...@@ -14,6 +14,8 @@
#include "git2/object.h" #include "git2/object.h"
#include "git2/oid.h" #include "git2/oid.h"
#include "git2/refs.h" #include "git2/refs.h"
#include "git2/tag.h"
#include "git2/commit.h"
GIT_BEGIN_DECL GIT_BEGIN_DECL
...@@ -146,7 +148,14 @@ static git_object* dereference_object(git_object *obj) ...@@ -146,7 +148,14 @@ static git_object* dereference_object(git_object *obj)
break; break;
case GIT_OBJ_REF_DELTA: case GIT_OBJ_REF_DELTA:
break; break;
}}
default:
break;
}
/* Can't dereference some types */
return NULL;
}
static int dereference_to_type(git_object **out, git_object *obj, git_otype target_type) static int dereference_to_type(git_object **out, git_object *obj, git_otype target_type)
{ {
...@@ -158,6 +167,11 @@ static int dereference_to_type(git_object **out, git_object *obj, git_otype targ ...@@ -158,6 +167,11 @@ static int dereference_to_type(git_object **out, git_object *obj, git_otype targ
return 0; return 0;
} }
if (this_type == GIT_OBJ_TAG) {
git_tag_peel(&obj, (git_tag*)obj);
continue;
}
/* Dereference once, if possible. */ /* Dereference once, if possible. */
obj = dereference_object(obj); obj = dereference_object(obj);
...@@ -167,8 +181,8 @@ static int dereference_to_type(git_object **out, git_object *obj, git_otype targ ...@@ -167,8 +181,8 @@ static int dereference_to_type(git_object **out, git_object *obj, git_otype targ
static int handle_caret_syntax(git_object **out, git_object *start, const char *movement) static int handle_caret_syntax(git_object **out, git_object *start, const char *movement)
{ {
git_object *obj; git_object *obj;
git_commit *commit;
printf("Moving by '%s'\n", movement); int n;
if (*movement == '{') { if (*movement == '{') {
if (movement[strlen(movement)-1] != '}') { if (movement[strlen(movement)-1] != '}') {
...@@ -184,10 +198,29 @@ static int handle_caret_syntax(git_object **out, git_object *start, const char * ...@@ -184,10 +198,29 @@ static int handle_caret_syntax(git_object **out, git_object *start, const char *
/* Dereference until we reach a commit. */ /* Dereference until we reach a commit. */
if (dereference_to_type(&obj, start, GIT_OBJ_COMMIT) < 0) { if (dereference_to_type(&obj, start, GIT_OBJ_COMMIT) < 0) {
/* Can't dereference to a commit; fail */
return GIT_ERROR;
} }
/* Move to the Nth parent. */ /* "^" is the same as "^1" */
if (strlen(movement) == 0) {
n = 1;
} else {
git__strtol32(&n, movement, NULL, 0);
}
commit = (git_commit*)obj;
/* "^0" just returns the input */
if (n == 0) {
*out = (git_object*)commit;
return 0;
}
if (git_commit_parent(&commit, commit, n-1) < 0) {
return GIT_ERROR;
}
*out = (git_object*)commit;
return 0; return 0;
} }
...@@ -197,6 +230,7 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec ...@@ -197,6 +230,7 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec
revparse_state next_state = REVPARSE_STATE_INIT; revparse_state next_state = REVPARSE_STATE_INIT;
const char *spec_cur = spec; const char *spec_cur = spec;
git_object *cur_obj = NULL; git_object *cur_obj = NULL;
git_object *next_obj = NULL;
git_buf specbuffer = GIT_BUF_INIT; git_buf specbuffer = GIT_BUF_INIT;
git_buf stepbuffer = GIT_BUF_INIT; git_buf stepbuffer = GIT_BUF_INIT;
int retcode = 0; int retcode = 0;
...@@ -210,6 +244,7 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec ...@@ -210,6 +244,7 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec
/* No operators, just a name. Find it and return. */ /* No operators, just a name. Find it and return. */
return revparse_lookup_object(out, repo, spec); return revparse_lookup_object(out, repo, spec);
} else if (*spec_cur == '@') { } else if (*spec_cur == '@') {
/* '@' syntax doesn't allow chaining */
git_buf_puts(&stepbuffer, spec_cur); git_buf_puts(&stepbuffer, spec_cur);
retcode = walk_ref_history(out, git_buf_cstr(&specbuffer), git_buf_cstr(&stepbuffer)); retcode = walk_ref_history(out, git_buf_cstr(&specbuffer), git_buf_cstr(&stepbuffer));
goto cleanup; goto cleanup;
...@@ -224,7 +259,7 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec ...@@ -224,7 +259,7 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec
if (current_state != next_state) { if (current_state != next_state) {
/* Leaving INIT state, find the object specified, in case that state needs it */ /* Leaving INIT state, find the object specified, in case that state needs it */
assert(!revparse_lookup_object(&cur_obj, repo, git_buf_cstr(&specbuffer))); assert(!revparse_lookup_object(&next_obj, repo, git_buf_cstr(&specbuffer)));
} }
break; break;
...@@ -237,15 +272,17 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec ...@@ -237,15 +272,17 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec
git_buf_cstr(&stepbuffer)); git_buf_cstr(&stepbuffer));
goto cleanup; goto cleanup;
} else if (*spec_cur == '~') { } else if (*spec_cur == '~') {
retcode = handle_caret_syntax(&cur_obj, retcode = handle_caret_syntax(&next_obj,
cur_obj, cur_obj,
git_buf_cstr(&stepbuffer)); git_buf_cstr(&stepbuffer));
git_buf_clear(&stepbuffer);
if (retcode < 0) goto cleanup; if (retcode < 0) goto cleanup;
next_state = REVPARSE_STATE_LINEAR; next_state = REVPARSE_STATE_LINEAR;
} else if (*spec_cur == '^') { } else if (*spec_cur == '^') {
retcode = handle_caret_syntax(&cur_obj, retcode = handle_caret_syntax(&next_obj,
cur_obj, cur_obj,
git_buf_cstr(&stepbuffer)); git_buf_cstr(&stepbuffer));
git_buf_clear(&stepbuffer);
if (retcode < 0) goto cleanup; if (retcode < 0) goto cleanup;
next_state = REVPARSE_STATE_CARET; next_state = REVPARSE_STATE_CARET;
} else { } else {
...@@ -259,6 +296,10 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec ...@@ -259,6 +296,10 @@ int git_revparse_single(git_object **out, git_repository *repo, const char *spec
} }
current_state = next_state; current_state = next_state;
if (cur_obj != next_obj) {
git_object_free(cur_obj);
cur_obj = next_obj;
}
} }
cleanup: cleanup:
......
...@@ -76,12 +76,16 @@ void test_refs_revparse__nth_parent(void) ...@@ -76,12 +76,16 @@ void test_refs_revparse__nth_parent(void)
{ {
cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^1")); cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^1"));
oid_str_cmp(git_object_id(g_obj), "9fd738e8f7967c078dceed8190330fc8648ee56a"); oid_str_cmp(git_object_id(g_obj), "9fd738e8f7967c078dceed8190330fc8648ee56a");
cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^"));
oid_str_cmp(git_object_id(g_obj), "9fd738e8f7967c078dceed8190330fc8648ee56a");
cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^2")); cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^2"));
oid_str_cmp(git_object_id(g_obj), "c47800c7266a2be04c571c04d5a6614691ea99bd"); oid_str_cmp(git_object_id(g_obj), "c47800c7266a2be04c571c04d5a6614691ea99bd");
cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^1^1")); cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^1^1"));
oid_str_cmp(git_object_id(g_obj), "4a202b346bb0fb0db7eff3cffeb3c70babbd2045"); oid_str_cmp(git_object_id(g_obj), "4a202b346bb0fb0db7eff3cffeb3c70babbd2045");
cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^2^1")); cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^2^1"));
oid_str_cmp(git_object_id(g_obj), "5b5b025afb0b4c913b4c338a42934a3863bf3644"); oid_str_cmp(git_object_id(g_obj), "5b5b025afb0b4c913b4c338a42934a3863bf3644");
cl_git_pass(git_revparse_single(&g_obj, g_repo, "be3563a^0"));
oid_str_cmp(git_object_id(g_obj), "be3563ae3f795b2b4353bcce3a527ad0a4f7f644");
} }
void test_refs_revparse__reflog(void) void test_refs_revparse__reflog(void)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment