之前写的一篇有些bug,而且里面的概念也没搞清楚,查询了《Git权威指南》后又优化了一下,如下:
public class GitUtil {
private final static String GIT = ".git";
private final static String REF_REMOTES = "refs/remotes/origin/";
/**
* 将文件列表提交到git仓库中
* @param gitRoot git仓库目录
* @param files 需要提交的文件列表
* @param remark 备注
* @return 返回本次提交的版本号
* @throws IOException
*/
public static String commitToGitRepository(String gitRoot, List<String> files, String remark)
throws Exception {
if (StringUtils.isNotBlank(gitRoot) && files != null && files.size() > 0) {
File rootDir = new File(gitRoot);
//初始化git仓库
if (new File(gitRoot + File.separator + GIT).exists() == false) {
Git.init().setDirectory(rootDir).call();
}
//打开git仓库
Git git = Git.open(rootDir);
//判断工作区与暂存区的文件内容是否有变更
List<DiffEntry> diffEntries = git.diff()
.setPathFilter(PathFilterGroup.createFromStrings(files))
.setShowNameAndStatusOnly(true).call();
if (diffEntries == null || diffEntries.size() == 0) {
throw new Exception("提交的文件内容都没有被修改,不能提交");
}
//被修改过的文件
List<String> updateFiles = new ArrayList<String>();
ChangeType changeType;
for (DiffEntry entry : diffEntries) {
changeType = entry.getChangeType();
switch (changeType) {
case ADD:
case COPY:
case RENAME:
case MODIFY:
updateFiles.add(entry.getNewPath());
break;
case DELETE:
updateFiles.add(entry.getOldPath());
break;
}
}
//将文件提交到git仓库中,并返回本次提交的版本号
//1、将工作区的内容更新到暂存区
AddCommand addCmd = git.add();
for (String file : updateFiles) {
addCmd.addFilepattern(file);
}
addCmd.call();
//2、commit
CommitCommand commitCmd = git.commit();
for (String file : updateFiles) {
commitCmd.setOnly(file);
}
RevCommit revCommit = commitCmd.setCommitter("yonge", "654166020@qq.com")
.setMessage(remark).call();
return revCommit.getName();
}
return null;
}
/**
* 回滚到指定版本的上一个版本
* @param gitRoot git仓库目录
* @param diffEntries 需要回滚的文件
* @param revision 版本号
* @param remark 备注
* @return
* @throws Exception
*/
public static boolean rollBackPreRevision(String gitRoot, List<DiffEntry> diffEntries,
String revision, String remark) throws Exception {
if (diffEntries == null || diffEntries.size() == 0) {
throw new Exception("没有需要回滚的文件");
}
Git git = Git.open(new File(gitRoot));
List<String> files = new ArrayList<String>();
//注意:下面的reset命令会将暂存区的内容恢复到指定(revesion)的状态,相当于取消add命令的操作
/*Repository repository = git.getRepository();
RevWalk walk = new RevWalk(repository);
ObjectId objId = repository.resolve(revision);
RevCommit revCommit = walk.parseCommit(objId);
String preVision = revCommit.getParent(0).getName();
ResetCommand resetCmd = git.reset();
for (String file : files) {
resetCmd.addPath(file);
}
resetCmd.setRef(preVision).call();
repository.close();*/
//取出需要回滚的文件,新增的文件不回滚
for (DiffEntry diffEntry : diffEntries) {
if (diffEntry.getChangeType() == ChangeType.DELETE) {
continue;
} else {
files.add(diffEntry.getNewPath());
}
}
if (files.size() == 0) {
throw new Exception("没有需要回滚的文件");
}
//checkout操作会丢失工作区的数据,暂存区和工作区的数据会恢复到指定(revision)的版本内容
CheckoutCommand checkoutCmd = git.checkout();
for (String file : files) {
checkoutCmd.addPath(file);
}
//加了“^”表示指定版本的前一个版本,如果没有上一版本,在命令行中会报错,例如:error: pathspec '4.vm' did not match any file(s) known to git.
checkoutCmd.setStartPoint(revision + "^");
checkoutCmd.call();
//重新提交一次
CommitCommand commitCmd = git.commit();
for (String file : files) {
commitCmd.setOnly(file);
}
commitCmd.setCommitter("yonge", "654166020@qq.com").setMessage(remark).call();
return true;
}
/**
* 获取上一版本的变更记录,如果是新增的文件,不会显示,因为做回滚时不需要回滚新增的文件
* @param gitRoot git仓库目录
* @param revision 版本号
* @return
* @throws Exception
*/
public static List<DiffEntry> rollBackFile(String gitRoot, String revision) throws Exception {
Git git = Git.open(new File(gitRoot));
Repository repository = git.getRepository();
ObjectId objId = repository.resolve(revision);
Iterable<RevCommit> allCommitsLater = git.log().add(objId).call();
Iterator<RevCommit> iter = allCommitsLater.iterator();
RevCommit commit = iter.next();
TreeWalk tw = new TreeWalk(repository);
tw.addTree(commit.getTree());
commit = iter.next();
if (commit != null) {
tw.addTree(commit.getTree());
} else {
throw new Exception("当前库只有一个版本,不能获取变更记录");
}
tw.setRecursive(true);
RenameDetector rd = new RenameDetector(repository);
rd.addAll(DiffEntry.scan(tw));
List<DiffEntry> diffEntries = rd.compute();
if (diffEntries == null || diffEntries.size() == 0) {
return diffEntries;
}
Iterator<DiffEntry> iterator = new ArrayList<DiffEntry>(diffEntries).iterator();
DiffEntry diffEntry = null;
while (iterator.hasNext()) {
diffEntry = iterator.next();
System.out.println("newPath:" + diffEntry.getNewPath() + " oldPath:"
+ diffEntry.getOldPath() + " changeType:"
+ diffEntry.getChangeType());
if (diffEntry.getChangeType() == ChangeType.DELETE) {
iterator.remove();
}
}
return diffEntries;
}
}
可能里面还会有一些问题,若发现请留言,谢谢!!
分享到:
相关推荐
JGit是Java实现的一个分布式版本控制系统,它提供了一个命令行接口和API,使得开发者可以在Java应用程序中集成版本控制功能。这个压缩包包含了JGit的主要库文件以及相关的文档资源。 首先,我们关注`org.eclipse....
综上所述,这个"JGit工具类包"提供了一整套的Git操作功能,涵盖了从基本的仓库管理和版本控制到复杂的GitLab API交互,对于开发Java应用尤其是需要集成Git功能的应用非常有用。`giylabapi`和递归操作的特性使得它在...
JGit是一个可以用java来对git进行操作的插件,可以通过代码连接git远程仓库,在本地建立版本库,进行拉取、提交、推送等操作。使用时添加pom依赖 <groupId>org.eclipse.jgit <artifactId>org.eclipse.jgit ...
将JGit与MATLAB结合,意味着MATLAB用户可以方便地在编程环境中执行Git的版本控制任务,如克隆、提交、拉取、合并等。 描述中提到,“jgit4matlab是jgit在matlab中的包装器”,这表明存在一个MATLAB接口或类库,使得...
赠送jar包:org.eclipse.jgit-5.10.0.202012080955-r.jar; 赠送原API文档:org.eclipse.jgit-5.10.0.202012080955-r-javadoc.jar; 赠送源代码:org.eclipse.jgit-5.10.0.202012080955-r-sources.jar; 赠送Maven...
Eclipse JGit是一款用Java编写的轻量级、纯Java库,它实现了Git版本控制系统的主要功能,使得在Java环境中可以进行Git操作,如克隆、提交、分支管理和合并等。 【描述】"scala-amazon-queue.zip" 提供了一组Scala...
org.eclipse.jgit-4.5.0.201609210915-r.jar 通过java执行git命令
赠送jar包:org.eclipse.jgit.lfs-5.10.0.202012080955-r.jar; 赠送原API文档:org.eclipse.jgit.lfs-5.10.0.202012080955-r-javadoc.jar; 赠送源代码:org.eclipse.jgit.lfs-5.10.0.202012080955-r-sources.jar;...
赠送jar包:org.eclipse.jgit-5.10.0.202012080955-r.jar; 赠送原API文档:org.eclipse.jgit-5.10.0.202012080955-r-javadoc.jar; 赠送源代码:org.eclipse.jgit-5.10.0.202012080955-r-sources.jar; 赠送Maven...
silly-jgit 是 Git 核心库的纯 Java 实现。 标签:silly
在实际开发中,JGit的一个常见应用场景是在Java Web应用中集成版本控制功能,例如,用户可以通过Web界面提交代码,而服务端使用JGit处理这些提交,同步到远程仓库。另一个应用场景是在服务器端实现自动化代码部署,...
JGit 是提供了一套类似 Git 命令的 Java API,可以方便地在程序中进行 git 操作 jgit-4.8.0.201706111038-r.jar
jgit-cookbook, 提供 JGit Java Git实现的示例和代码 Fragment jgit食谱 提供 JGit的Java Git实现示例和代码 Fragment 。JGit框架丰富多样,它有两层,一个低级的插件 api,以及一个高级别的系列的命令,比如 ...
JGit 是提供了一套类似 Git 命令的 Java API,可以方便地在程序中进行 git 操作
3. **提交历史**:查询提交历史,获取特定提交的信息。 4. **差异计算**:比较文件或目录在不同版本之间的变化。 5. **标签管理**:添加、删除和查询Git标签。 6. **索引和工作树**:管理Git的工作树和索引,执行暂...
赠送jar包:org.eclipse.jgit.lfs-5.10.0.202012080955-r.jar; 赠送原API文档:org.eclipse.jgit.lfs-5.10.0.202012080955-r-javadoc.jar; 赠送源代码:org.eclipse.jgit.lfs-5.10.0.202012080955-r-sources.jar;...
JGit 是一个用 Java 编写的轻量级、纯Java实现的版本控制系统,它提供了与Git类似的功能,可以在没有Git原生支持的环境中使用。在本文中,我们将深入探讨JGit的基本概念、核心功能以及如何在Java项目中使用它。 **...
Eclipse JGit是一个用Java编写的轻量级、纯Java库,用于处理Git版本控制系统。这个特定的版本可能是开发者的非官方测试构建,版本号2.99.99.1.0表明它可能是一个非常规版本,用于测试或实验目的,由名为ROBERTO的...
Ant是Apache的一款构建工具,而JGit的Ant任务库则允许开发者在Ant构建脚本中直接使用JGit的功能,比如克隆、提交、拉取和分支管理等。这对于自动化构建流程非常有用,特别是对于那些依赖于Git版本控制的Java项目。 ...