2

git_stash_save()允许保存更改,类似于git stash. 回退有什么功能git stash pop吗?

我可以看到git_stash_foreach()git_stash_drop()。有没有办法使用它们来实现这个功能?


编辑:根据nulltoken 的回答,我预计以下代码可以工作:

void tstStashPop ( const char * repo_path )
{
  git_repository *repo;
  git_commit * top_cmt;
  git_oid saved_stash;
  git_tree * top_tree;
  git_signature *signature;

  // open a repository
  if ( git_repository_open(&repo, repo_path) != 0 )
  {
    assert(false);
  }
  else
  {
    // create a signature
    git_signature_new(&signature, "no name", "no.name@gmail.com", 1323847743, 60);

    if ( git_stash_save( &saved_stash, repo, signature,
               "message for this stash", /*GIT_STASH_INCLUDE_UNTRACKED*/0) 
       != GIT_ENOTFOUND )
    {
      // get the commit that was saved by git stash save
      if ( git_commit_lookup( &top_cmt, repo, &saved_stash ) != 0 ) 
      {
        assert(false);
      }
      else
      {
        // get the tree for this commit
        if ( git_commit_tree( &top_tree, top_cmt ) != 0 )
        {
          assert(false);
        }
        else
        {
          // checkout the tree
          git_checkout_opts opts;
          opts = GIT_CHECKOUT_OPTS_INIT;
          opts.checkout_strategy = GIT_CHECKOUT_SAFE_CREATE;
          if ( git_checkout_tree( repo, (git_object*)top_tree, &opts ) != 0 )
          {
            assert(false);
          }
        }
      }
      // remove the stashed commit
      git_stash_drop( repo, 0 );
    }

    // free signature
    git_signature_free(signature);

    // free repo
    git_repository_free(repo);
  }
}

没有报告错误,但未恢复更改。git_stash_save()有效(我可以看到消息git stash list)并且git_stash_drop()也有效。但是,git_checkout_tree()不会产生任何效果。

另外,我应该免费top_treetop_cmt

4

2 回答 2

2

是否有git stash将更改弹出的功能?

还没有。实际上,git stash (pop | apply)会将存储的内容与工作目录的当前内容合并。

不幸的是,在 libgit2 中尚不merge可用。

更新

但是, git_checkout_tree() 没有任何效果。

您可能希望通过opts.checkout_strategy. 默认值是试运行,不会更新任何内容。

您可以参考include/git2/checkout.h以获取有关选项的更多详细信息。

另外,我应该释放 top_tree 和 top_cmt 吗?

确实git_tree_free()git_commit_free()在这里会很有用。

或者,您可以避免调用并直接传递将其剥离到树中git_commit_tree()的提交。git_checkout_tree()

于 2013-03-29T09:44:43.263 回答
2

最终变体作为未来搜索者的复制粘贴片段:

void tstStashPop ( const char * repo_path )
{
  git_repository *repo;
  git_commit * top_cmt;
  git_oid saved_stash;
  git_tree * top_tree;
  git_signature *signature;

  // open a repository
  if ( git_repository_open(&repo, repo_path) != 0 )
  {
    assert(false);
  }
  else
  {
    // create a signature
    git_signature_new(&signature, "no name", "no.name@gmail.com", 1323847743, 60);

    if ( git_stash_save( &saved_stash, repo, signature,
               "message for this stash", /*GIT_STASH_INCLUDE_UNTRACKED*/0) 
       != GIT_ENOTFOUND )
    {
      // get the commit that was saved by git stash save
      if ( git_commit_lookup( &top_cmt, repo, &saved_stash ) != 0 ) 
      {
        assert(false);
      }
      else
      {
        // get the tree for this commit
        if ( git_commit_tree( &top_tree, top_cmt ) != 0 )
        {
          assert(false);
        }
        else
        {
          // checkout the tree
          git_checkout_opts opts;
          opts = GIT_CHECKOUT_OPTS_INIT;
          opts.checkout_strategy = GIT_CHECKOUT_SAFE_CREATE;
          if ( git_checkout_tree( repo, (git_object*)top_tree, &opts ) != 0 )
          {
            assert(false);
          }
          git_tree_free(top_tree);
        }
        git_commit_free(top_cmt);
      }
      // remove the stashed commit
      git_stash_drop( repo, 0 );
    }

    // free signature
    git_signature_free(signature);

    // free repo
    git_repository_free(repo);
  }
}
于 2013-03-29T14:23:01.383 回答