From 5a800efc42b5c58d06e0335348d75bde82af6ef7 Mon Sep 17 00:00:00 2001 From: Vicent Marti Date: Wed, 9 Feb 2011 12:46:54 +0200 Subject: [PATCH] Honor alternate entries in the ODB The alternates file is now parsed, and the alternate ODB folders are added as separate backends. This allows the library to efficiently query the alternate folders. Signed-off-by: Vicent Marti --- src/odb.c | 79 ++++++++++++++++++++++++++++++++++++++++++++----------- 1 file changed, 63 insertions(+), 16 deletions(-) diff --git a/src/odb.c b/src/odb.c index 26b457b7c..b5996c826 100644 --- a/src/odb.c +++ b/src/odb.c @@ -33,6 +33,8 @@ #include "git2/odb_backend.h" +#define GIT_ALTERNATES_FILE "info/alternates" + static int format_object_header(char *hdr, size_t n, git_rawobj *obj) { const char *type_str = git_object_type2string(obj->type); @@ -171,32 +173,77 @@ int git_odb_add_backend(git_odb *odb, git_odb_backend *backend) return GIT_SUCCESS; } +static int add_default_backends(git_odb *db, const char *objects_dir) +{ + git_odb_backend *loose, *packed; + int error; + + /* add the loose object backend */ + error = git_odb_backend_loose(&loose, objects_dir); + if (error < GIT_SUCCESS) + return error; + + error = git_odb_add_backend(db, loose); + if (error < GIT_SUCCESS) + return error; + + /* add the packed file backend */ + error = git_odb_backend_pack(&packed, objects_dir); + if (error < GIT_SUCCESS) + return error; + + error = git_odb_add_backend(db, packed); + if (error < GIT_SUCCESS) + return error; + + return GIT_SUCCESS; +} + +static int load_alternates(git_odb *odb, const char *objects_dir) +{ + char alternates_path[GIT_PATH_MAX]; + char alternate[GIT_PATH_MAX]; + char *buffer; + + gitfo_buf alternates_buf = GITFO_BUF_INIT; + int error; + + git__joinpath(alternates_path, objects_dir, GIT_ALTERNATES_FILE); + + if (gitfo_exists(alternates_path) < GIT_SUCCESS) + return GIT_SUCCESS; + + if (gitfo_read_file(&alternates_buf, alternates_path) < GIT_SUCCESS) + return GIT_EOSERR; + + buffer = (char *)alternates_buf.data; + error = GIT_SUCCESS; + + /* add each alternate as a new backend; one alternate per line */ + while ((error == GIT_SUCCESS) && (buffer = git__strtok(alternate, buffer, "\r\n")) != NULL) + error = add_default_backends(odb, alternate); + + gitfo_free_buf(&alternates_buf); + return error; +} int git_odb_open(git_odb **out, const char *objects_dir) { git_odb *db; - git_odb_backend *loose, *packed; int error; + assert(out && objects_dir); + + *out = NULL; + if ((error = git_odb_new(&db)) < 0) return error; - /* add the loose object backend */ - if (git_odb_backend_loose(&loose, objects_dir) == 0) { - error = git_odb_add_backend(db, loose); - if (error < 0) - goto cleanup; - } + if ((error = add_default_backends(db, objects_dir)) < GIT_SUCCESS) + goto cleanup; - /* add the packed file backend */ - if (git_odb_backend_pack(&packed, objects_dir) == 0) { - error = git_odb_add_backend(db, packed); - if (error < 0) - goto cleanup; - } - - /* TODO: add altenernates as new backends; - * how elevant is that? very elegant. */ + if ((error = load_alternates(db, objects_dir)) < GIT_SUCCESS) + goto cleanup; *out = db; return GIT_SUCCESS;