克隆一个Git仓库(深入)

9

如何使用 libgit2 克隆一个仓库

我想要做的就是像 git clone 一样,但是使用 libgit2。我可能在问 git clone 的详细操作。

目前我正在进行以下步骤:

  1. 初始化一个仓库
  2. 调整配置文件以添加远程仓库
  3. 创建一个 git_remote
  4. 下载一个 packfile
  5. 为 packfile 建立索引并写入索引(生成一个 .idx 文件)
  6. (编辑) 将所有不同的分支写入磁盘。
  7. (编辑) 以某种方式执行 git checkout

现在我不知道该怎么做了。我唯一的猜测是将 .idx 加载到 git_index 中并使用 git_repository_set_index,但这也没有显示任何文件。

编辑

我测试了在仅克隆一半的代码库上运行git checkout master,这就解决了问题。现在我只需要找出如何使用libgit2来做到这一点,并且问题跟踪器中似乎有一些有用的信息。 编辑2 我将添加我的当前代码,希望有一天有人会发现它有用,希望成为我从未找到的快速启动代码。注意:我在这里使用Obj-CObjective-Git,但主要是纯c。
+ (GTRepository *)cloneFromRemoteURL:(NSURL *)remoteURL toLocalURL:(NSURL *)localURL
{   
// Let's suppose the URL looks like: https://github.com/libgit2/libgit2.git
// Then we need to get a URL like this too: git://github.com/libgit2/libgit2.git
// This may be a bit dodgy, but it will do for now.
const char *gitURL = [remoteURL.absoluteString stringByReplacingOccurrencesOfString:@"https://github.com/" withString:@"git://github.com/"].UTF8String;

//Setup
int error;
git_repository *repo
git_config *cfg;
git_remote *remote;

NSURL *gitDirURL = [localURL URLByAppendingPathComponent:@".git"];

error = git_repository_open(&repo, gitDirURL.path.UTF8String);
    if (error != GIT_SUCCESS) {

    git_repository_init(&repo, gitDirURL.path.UTF8String, 1);

    //Config
    git_repository_config(&cfg, repo);
    git_config_set_int32 (cfg, "core.repositoryformatversion", 0);
    git_config_set_bool (cfg, "core.filemode", 1);
    git_config_set_bool (cfg, "core.bare", 0);
    git_config_set_bool (cfg, "core.logallrefupdates", 1);
    git_config_set_bool (cfg, "core.ignorecase", 1);
    git_config_set_string (cfg, "remote.origin.url", gitURL);
    git_config_set_string (cfg, "remote.origin.fetch", "+refs/heads/*:refs/remotes/origin/*");
    git_config_set_string (cfg, "branch.master.remote", "origin");
    git_config_set_string (cfg, "branch.master.merge", "refs/heads/master");

    git_repository_set_workdir(repo, localURL.path.UTF8String);

    error = git_remote_new(&remote, repo, "A remote", gitURL, "origin");

    git_repository_free(repo);
    git_repository_open (&repo, localURL.path.UTF8String);
}



git_repository_config(&cfg, repo);

// connect to repo
error = git_remote_load(&remote, repo, "origin");

error = git_remote_connect(remote, GIT_DIR_FETCH);
// get pack file

git_off_t bytes;
git_indexer_stats stats;
error = git_remote_download(remote, &bytes, &stats);

NSURL *packFolderURL = [localURL URLByAppendingPathComponent:@".git/objects/pack"];
NSFileManager *fileManager = [NSFileManager defaultManager];
NSArray *array = [fileManager contentsOfDirectoryAtURL:packFolderURL includingPropertiesForKeys:nil options:0 error:nil];
NSLog(@"Dictionary:%@",array);
NSString *result;
for (NSURL *url in array) {
    if ([url.path rangeOfString:@".pack"].location != NSNotFound) {
        result = url.path;
    }
}
const char *packname = [result UTF8String];


// unpack pack file
if (packname != NULL) 
{
    git_indexer *indexer;
    git_indexer_stats stats2;
    int error;
    char hash[GIT_OID_HEXSZ + 1] = {0};

    error = git_indexer_new(&indexer, packname);
    error = git_indexer_run(indexer, &stats2);
    error = git_indexer_write(indexer);

    // Get the packfile's hash (which should become it's filename)
    git_oid_fmt(hash, git_indexer_hash(indexer));

    NSString *hashStr = [NSString stringWithCString:hash encoding:NSUTF8StringEncoding];
    hashStr = [NSString stringWithFormat:@"pack-%@.idx",hashStr];
    const char *indexPath = [hashStr UTF8String];

    puts(hash);
    git_index *index;
    git_index_open(&index, indexPath);
    git_index_read(index);
    git_repository_set_index(repo, index);


    git_indexer_free(indexer);
    git_remote_update_tips(remote, update_cb2); //No idea what it does, but it seems like it's important… It does make the branches appear in .git/refs/remotes/origin

}

//在这里执行git checkout master命令

return [GTRepository repositoryWithURL:localURL error:nil];

}

有趣的反馈。这应该以某种方式集成到 gitlib2 库中。 - VonC
1个回答

4

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接