blob: 63ab8094ae92d864b62b1eda876c21af57487e8e [file] [log] [blame]
/*
* Copyright (C) 2011, 2022 Chris Aniszczyk <caniszczyk@gmail.com> and others
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Distribution License v. 1.0 which is available at
* https://www.eclipse.org/org/documents/edl-v10.php.
*
* SPDX-License-Identifier: BSD-3-Clause
*/
package org.eclipse.jgit.api;
import static org.junit.Assert.assertArrayEquals;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotEquals;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import java.io.File;
import java.io.IOException;
import java.net.URISyntaxException;
import java.time.Instant;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;
import org.eclipse.jgit.api.ListBranchCommand.ListMode;
import org.eclipse.jgit.api.errors.GitAPIException;
import org.eclipse.jgit.api.errors.JGitInternalException;
import org.eclipse.jgit.errors.NoWorkTreeException;
import org.eclipse.jgit.junit.RepositoryTestCase;
import org.eclipse.jgit.junit.TestRepository;
import org.eclipse.jgit.lib.BranchConfig.BranchRebaseMode;
import org.eclipse.jgit.lib.ConfigConstants;
import org.eclipse.jgit.lib.Constants;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.RefUpdate;
import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.lib.StoredConfig;
import org.eclipse.jgit.revwalk.RevBlob;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevObject;
import org.eclipse.jgit.submodule.SubmoduleStatus;
import org.eclipse.jgit.submodule.SubmoduleStatusType;
import org.eclipse.jgit.submodule.SubmoduleWalk;
import org.eclipse.jgit.transport.RefSpec;
import org.eclipse.jgit.transport.RemoteConfig;
import org.eclipse.jgit.transport.TagOpt;
import org.eclipse.jgit.transport.URIish;
import org.eclipse.jgit.util.SystemReader;
import org.junit.Test;
public class CloneCommandTest extends RepositoryTestCase {
private Git git;
private TestRepository<Repository> tr;
@Override
public void setUp() throws Exception {
super.setUp();
tr = new TestRepository<>(db);
git = new Git(db);
// commit something
writeTrashFile("Test.txt", "Hello world");
git.add().addFilepattern("Test.txt").call();
git.commit().setMessage("Initial commit").call();
Ref head = git.tag().setName("tag-initial").setMessage("Tag initial")
.call();
// create a test branch and switch to it
git.checkout().setCreateBranch(true).setName("test").call();
// create a non-standard ref
RefUpdate ru = db.updateRef("refs/meta/foo/bar");
ru.setNewObjectId(head.getObjectId());
ru.update();
// commit something on the test branch
writeTrashFile("Test.txt", "Some change");
git.add().addFilepattern("Test.txt").call();
git.commit().setMessage("Second commit").call();
RevBlob blob = tr.blob("blob-not-in-master-branch");
git.tag().setName("tag-for-blob").setObjectId(blob).call();
}
@Test
public void testCloneRepository() throws IOException,
JGitInternalException, GitAPIException, URISyntaxException {
File directory = createTempDirectory("testCloneRepository");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
ObjectId id = git2.getRepository().resolve("tag-for-blob");
assertNotNull(id);
assertEquals(git2.getRepository().getFullBranch(), "refs/heads/test");
assertEquals(
"origin",
git2.getRepository()
.getConfig()
.getString(ConfigConstants.CONFIG_BRANCH_SECTION,
"test", ConfigConstants.CONFIG_KEY_REMOTE));
assertEquals(
"refs/heads/test",
git2.getRepository()
.getConfig()
.getString(ConfigConstants.CONFIG_BRANCH_SECTION,
"test", ConfigConstants.CONFIG_KEY_MERGE));
assertEquals(2, git2.branchList().setListMode(ListMode.REMOTE).call()
.size());
assertEquals(new RefSpec("+refs/heads/*:refs/remotes/origin/*"),
fetchRefSpec(git2.getRepository()));
assertTagOption(git2.getRepository(), TagOpt.AUTO_FOLLOW);
}
@Test
public void testCloneRepositoryNoCheckout()
throws IOException, JGitInternalException, GitAPIException {
File directory = createTempDirectory("testCloneRepositoryNoCheckout");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setNoCheckout(true);
try (Git git2 = command.call()) {
Repository clonedRepo = git2.getRepository();
Ref main = clonedRepo.exactRef(Constants.R_HEADS + "test");
assertNotNull(main);
ObjectId id = main.getObjectId();
assertNotNull(id);
assertNotEquals(id, ObjectId.zeroId());
ObjectId headId = clonedRepo.resolve(Constants.HEAD);
assertEquals(id, headId);
assertArrayEquals(new String[] { Constants.DOT_GIT },
directory.list());
}
}
@Test
public void testCloneRepositoryRefLogForLocalRefs()
throws IOException, JGitInternalException, GitAPIException {
File directory = createTempDirectory(
"testCloneRepositoryRefLogForLocalRefs");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
Repository clonedRepo = git2.getRepository();
addRepoToClose(clonedRepo);
List<Ref> clonedRefs = clonedRepo.getRefDatabase().getRefs();
Stream<Ref> remoteRefs = clonedRefs.stream()
.filter(CloneCommandTest::isRemote);
Stream<Ref> localHeadsRefs = clonedRefs.stream()
.filter(CloneCommandTest::isLocalHead);
remoteRefs.forEach(ref -> assertFalse(
"Ref " + ref.getName()
+ " is remote and should not have a reflog",
hasRefLog(clonedRepo, ref)));
localHeadsRefs.forEach(ref -> assertTrue(
"Ref " + ref.getName()
+ " is local head and should have a reflog",
hasRefLog(clonedRepo, ref)));
}
private static boolean isRemote(Ref ref) {
return ref.getName().startsWith(Constants.R_REMOTES);
}
private static boolean isLocalHead(Ref ref) {
return !isRemote(ref) && ref.getName().startsWith(Constants.R_HEADS);
}
private static boolean hasRefLog(Repository repo, Ref ref) {
try {
return repo.getReflogReader(ref.getName()).getLastEntry() != null;
} catch (IOException ioe) {
throw new IllegalStateException(ioe);
}
}
@Test
public void testCloneRepositoryExplicitGitDir() throws IOException,
JGitInternalException, GitAPIException {
File directory = createTempDirectory("testCloneRepository");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setGitDir(new File(directory, Constants.DOT_GIT));
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals(directory, git2.getRepository().getWorkTree());
assertEquals(new File(directory, Constants.DOT_GIT), git2.getRepository()
.getDirectory());
}
@Test
public void testCloneRepositoryDefaultDirectory()
throws URISyntaxException, JGitInternalException {
CloneCommand command = Git.cloneRepository().setURI(fileUri());
command.verifyDirectories(new URIish(fileUri()));
File directory = command.getDirectory();
assertEquals(git.getRepository().getWorkTree().getName(), directory.getName());
}
@Test
public void testCloneBareRepositoryDefaultDirectory()
throws URISyntaxException, JGitInternalException {
CloneCommand command = Git.cloneRepository().setURI(fileUri()).setBare(true);
command.verifyDirectories(new URIish(fileUri()));
File directory = command.getDirectory();
assertEquals(git.getRepository().getWorkTree().getName() + Constants.DOT_GIT_EXT, directory.getName());
}
@Test
public void testCloneRepositoryExplicitGitDirNonStd() throws IOException,
JGitInternalException, GitAPIException {
File directory = createTempDirectory("testCloneRepository");
File gDir = createTempDirectory("testCloneRepository.git");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setGitDir(gDir);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals(directory, git2.getRepository().getWorkTree());
assertEquals(gDir, git2.getRepository()
.getDirectory());
assertTrue(new File(directory, Constants.DOT_GIT).isFile());
assertFalse(new File(gDir, Constants.DOT_GIT).exists());
}
@Test
public void testCloneRepositoryExplicitGitDirBare() throws IOException,
JGitInternalException, GitAPIException {
File gDir = createTempDirectory("testCloneRepository.git");
CloneCommand command = Git.cloneRepository();
command.setBare(true);
command.setGitDir(gDir);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
try {
assertNull(null, git2.getRepository().getWorkTree());
fail("Expected NoWorkTreeException");
} catch (NoWorkTreeException e) {
assertEquals(gDir, git2.getRepository().getDirectory());
}
}
@Test
public void testBareCloneRepository() throws IOException,
JGitInternalException, GitAPIException, URISyntaxException {
File directory = createTempDirectory("testCloneRepository_bare");
CloneCommand command = Git.cloneRepository();
command.setBare(true);
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals(new RefSpec("+refs/heads/*:refs/heads/*"),
fetchRefSpec(git2.getRepository()));
}
@Test
public void testCloneRepositoryCustomRemote() throws Exception {
File directory = createTempDirectory("testCloneRemoteUpstream");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setRemote("upstream");
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("+refs/heads/*:refs/remotes/upstream/*",
git2.getRepository()
.getConfig()
.getStringList("remote", "upstream",
"fetch")[0]);
assertEquals("upstream",
git2.getRepository()
.getConfig()
.getString("branch", "test", "remote"));
assertEquals(db.resolve("test"),
git2.getRepository().resolve("upstream/test"));
}
@Test
public void testBareCloneRepositoryCustomRemote() throws Exception {
File directory = createTempDirectory("testCloneRemoteUpstream_bare");
CloneCommand command = Git.cloneRepository();
command.setBare(true);
command.setDirectory(directory);
command.setRemote("upstream");
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("+refs/heads/*:refs/heads/*",
git2.getRepository()
.getConfig()
.getStringList("remote", "upstream",
"fetch")[0]);
assertEquals("upstream",
git2.getRepository()
.getConfig()
.getString("branch", "test", "remote"));
assertNull(git2.getRepository().resolve("upstream/test"));
}
@Test
public void testBareCloneRepositoryNullRemote() throws Exception {
File directory = createTempDirectory("testCloneRemoteNull_bare");
CloneCommand command = Git.cloneRepository();
command.setBare(true);
command.setDirectory(directory);
command.setRemote(null);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("+refs/heads/*:refs/heads/*", git2.getRepository()
.getConfig().getStringList("remote", "origin", "fetch")[0]);
assertEquals("origin", git2.getRepository().getConfig()
.getString("branch", "test", "remote"));
}
public static RefSpec fetchRefSpec(Repository r) throws URISyntaxException {
RemoteConfig remoteConfig =
new RemoteConfig(r.getConfig(), Constants.DEFAULT_REMOTE_NAME);
return remoteConfig.getFetchRefSpecs().get(0);
}
@Test
public void testCloneRepositoryWithBranch() throws IOException,
JGitInternalException, GitAPIException {
File directory = createTempDirectory("testCloneRepositoryWithBranch");
CloneCommand command = Git.cloneRepository();
command.setBranch("refs/heads/master");
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("refs/heads/master", git2.getRepository().getFullBranch());
assertEquals(
"refs/heads/master, refs/remotes/origin/master, refs/remotes/origin/test",
allRefNames(git2.branchList().setListMode(ListMode.ALL).call()));
// Same thing, but now without checkout
directory = createTempDirectory(
"testCloneRepositoryWithBranch_noCheckout");
command = Git.cloneRepository();
command.setBranch("refs/heads/master");
command.setDirectory(directory);
command.setURI(fileUri());
command.setNoCheckout(true);
git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals(git2.getRepository().getFullBranch(), "refs/heads/master");
assertEquals(
"refs/heads/master, refs/remotes/origin/master, refs/remotes/origin/test",
allRefNames(git2.branchList().setListMode(ListMode.ALL).call()));
// Same thing, but now test with bare repo
directory = createTempDirectory("testCloneRepositoryWithBranch_bare");
command = Git.cloneRepository();
command.setBranch("refs/heads/master");
command.setDirectory(directory);
command.setURI(fileUri());
command.setBare(true);
git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("refs/heads/master", git2.getRepository().getFullBranch());
assertEquals("refs/heads/master, refs/heads/test", allRefNames(git2
.branchList().setListMode(ListMode.ALL).call()));
}
@Test
public void testCloneRepositoryWithBranchShortName() throws Exception {
File directory = createTempDirectory("testCloneRepositoryWithBranch");
CloneCommand command = Git.cloneRepository();
command.setBranch("test");
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("refs/heads/test", git2.getRepository().getFullBranch());
}
@Test
public void testCloneRepositoryWithTagName() throws Exception {
File directory = createTempDirectory("testCloneRepositoryWithBranch");
CloneCommand command = Git.cloneRepository();
command.setBranch("tag-initial");
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
ObjectId taggedCommit = db.resolve("tag-initial^{commit}");
assertEquals(taggedCommit.name(), git2
.getRepository().getFullBranch());
}
@Test
public void testCloneRepositoryOnlyOneBranch() throws Exception {
File directory = createTempDirectory("testCloneRepositoryWithBranch");
CloneCommand command = Git.cloneRepository();
command.setBranch("refs/heads/master");
command.setBranchesToClone(Collections
.singletonList("refs/heads/master"));
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertNull(git2.getRepository().resolve("tag-for-blob"));
assertNotNull(git2.getRepository().resolve("tag-initial"));
assertEquals("refs/heads/master", git2.getRepository().getFullBranch());
assertEquals("refs/remotes/origin/master", allRefNames(git2
.branchList().setListMode(ListMode.REMOTE).call()));
RemoteConfig cfg = new RemoteConfig(git2.getRepository().getConfig(),
Constants.DEFAULT_REMOTE_NAME);
List<RefSpec> specs = cfg.getFetchRefSpecs();
assertEquals(1, specs.size());
assertEquals(
new RefSpec("+refs/heads/master:refs/remotes/origin/master"),
specs.get(0));
}
@Test
public void testBareCloneRepositoryOnlyOneBranch() throws Exception {
File directory = createTempDirectory(
"testCloneRepositoryWithBranch_bare");
CloneCommand command = Git.cloneRepository();
command.setBranch("refs/heads/master");
command.setBranchesToClone(Collections
.singletonList("refs/heads/master"));
command.setDirectory(directory);
command.setURI(fileUri());
command.setBare(true);
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertNull(git2.getRepository().resolve("tag-for-blob"));
assertNotNull(git2.getRepository().resolve("tag-initial"));
assertEquals("refs/heads/master", git2.getRepository().getFullBranch());
assertEquals("refs/heads/master", allRefNames(git2.branchList()
.setListMode(ListMode.ALL).call()));
RemoteConfig cfg = new RemoteConfig(git2.getRepository().getConfig(),
Constants.DEFAULT_REMOTE_NAME);
List<RefSpec> specs = cfg.getFetchRefSpecs();
assertEquals(1, specs.size());
assertEquals(
new RefSpec("+refs/heads/master:refs/heads/master"),
specs.get(0));
}
@Test
public void testBareCloneRepositoryMirror() throws Exception {
File directory = createTempDirectory(
"testCloneRepositoryWithBranch_mirror");
CloneCommand command = Git.cloneRepository();
command.setBranch("refs/heads/master");
command.setMirror(true); // implies bare repository
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertTrue(git2.getRepository().isBare());
assertNotNull(git2.getRepository().resolve("tag-for-blob"));
assertNotNull(git2.getRepository().resolve("tag-initial"));
assertEquals("refs/heads/master", git2.getRepository().getFullBranch());
assertEquals("refs/heads/master, refs/heads/test", allRefNames(
git2.branchList().setListMode(ListMode.ALL).call()));
assertNotNull(git2.getRepository().exactRef("refs/meta/foo/bar"));
RemoteConfig cfg = new RemoteConfig(git2.getRepository().getConfig(),
Constants.DEFAULT_REMOTE_NAME);
List<RefSpec> specs = cfg.getFetchRefSpecs();
assertEquals(1, specs.size());
assertEquals(new RefSpec("+refs/*:refs/*"),
specs.get(0));
}
@Test
public void testCloneRepositoryOnlyOneTag() throws Exception {
File directory = createTempDirectory("testCloneRepositoryWithBranch");
CloneCommand command = Git.cloneRepository();
command.setBranch("tag-initial");
command.setBranchesToClone(
Collections.singletonList("refs/tags/tag-initial"));
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertNull(git2.getRepository().resolve("tag-for-blob"));
assertNull(git2.getRepository().resolve("refs/heads/master"));
assertNotNull(git2.getRepository().resolve("tag-initial"));
ObjectId taggedCommit = db.resolve("tag-initial^{commit}");
assertEquals(taggedCommit.name(), git2.getRepository().getFullBranch());
RemoteConfig cfg = new RemoteConfig(git2.getRepository().getConfig(),
Constants.DEFAULT_REMOTE_NAME);
List<RefSpec> specs = cfg.getFetchRefSpecs();
assertEquals(1, specs.size());
assertEquals(
new RefSpec("+refs/tags/tag-initial:refs/tags/tag-initial"),
specs.get(0));
}
@Test
public void testCloneRepositoryAllBranchesTakesPreference()
throws Exception {
File directory = createTempDirectory(
"testCloneRepositoryAllBranchesTakesPreference");
CloneCommand command = Git.cloneRepository();
command.setCloneAllBranches(true);
command.setBranchesToClone(
Collections.singletonList("refs/heads/test"));
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("refs/heads/test", git2.getRepository().getFullBranch());
// Expect both remote branches to exist; setCloneAllBranches(true)
// should override any setBranchesToClone().
assertNotNull(
git2.getRepository().resolve("refs/remotes/origin/master"));
assertNotNull(git2.getRepository().resolve("refs/remotes/origin/test"));
RemoteConfig cfg = new RemoteConfig(git2.getRepository().getConfig(),
Constants.DEFAULT_REMOTE_NAME);
List<RefSpec> specs = cfg.getFetchRefSpecs();
assertEquals(1, specs.size());
assertEquals(new RefSpec("+refs/heads/*:refs/remotes/origin/*"),
specs.get(0));
}
@Test
public void testCloneRepositoryAllBranchesIndependent() throws Exception {
File directory = createTempDirectory(
"testCloneRepositoryAllBranchesIndependent");
CloneCommand command = Git.cloneRepository();
command.setCloneAllBranches(true);
command.setBranchesToClone(
Collections.singletonList("refs/heads/test"));
command.setCloneAllBranches(false);
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("refs/heads/test", git2.getRepository().getFullBranch());
// Expect only the test branch; allBranches was re-set to false
assertNull(git2.getRepository().resolve("refs/remotes/origin/master"));
assertNotNull(git2.getRepository().resolve("refs/remotes/origin/test"));
RemoteConfig cfg = new RemoteConfig(git2.getRepository().getConfig(),
Constants.DEFAULT_REMOTE_NAME);
List<RefSpec> specs = cfg.getFetchRefSpecs();
assertEquals(1, specs.size());
assertEquals(new RefSpec("+refs/heads/test:refs/remotes/origin/test"),
specs.get(0));
}
public static String allRefNames(List<Ref> refs) {
StringBuilder sb = new StringBuilder();
for (Ref f : refs) {
if (sb.length() > 0)
sb.append(", ");
sb.append(f.getName());
}
return sb.toString();
}
@Test
public void testCloneRepositoryWhenDestinationDirectoryExistsAndIsNotEmpty()
throws IOException, JGitInternalException, GitAPIException {
String dirName = "testCloneTargetDirectoryNotEmpty";
File directory = createTempDirectory(dirName);
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
// clone again
command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
try {
git2 = command.call();
// we shouldn't get here
fail("destination directory already exists and is not an empty folder, cloning should fail");
} catch (JGitInternalException e) {
assertTrue(e.getMessage().contains("not an empty directory"));
assertTrue(e.getMessage().contains(dirName));
}
}
@Test
public void testCloneRepositoryWithMultipleHeadBranches() throws Exception {
git.checkout().setName(Constants.MASTER).call();
git.branchCreate().setName("a").call();
File directory = createTempDirectory("testCloneRepositoryWithMultipleHeadBranches");
CloneCommand clone = Git.cloneRepository();
clone.setDirectory(directory);
clone.setURI(fileUri());
Git git2 = clone.call();
addRepoToClose(git2.getRepository());
assertEquals(Constants.MASTER, git2.getRepository().getBranch());
}
@Test
public void testCloneRepositoryWithSubmodules() throws Exception {
git.checkout().setName(Constants.MASTER).call();
String file = "file.txt";
writeTrashFile(file, "content");
git.add().addFilepattern(file).call();
RevCommit commit = git.commit().setMessage("create file").call();
SubmoduleAddCommand command = new SubmoduleAddCommand(db);
String path = "sub";
command.setPath(path);
String uri = db.getDirectory().toURI().toString();
command.setURI(uri);
Repository repo = command.call();
assertNotNull(repo);
addRepoToClose(repo);
git.add().addFilepattern(path)
.addFilepattern(Constants.DOT_GIT_MODULES).call();
git.commit().setMessage("adding submodule").call();
try (SubmoduleWalk walk = SubmoduleWalk.forIndex(git.getRepository())) {
assertTrue(walk.next());
Repository subRepo = walk.getRepository();
addRepoToClose(subRepo);
assertNotNull(subRepo);
assertEquals(
new File(git.getRepository().getWorkTree(), walk.getPath()),
subRepo.getWorkTree());
assertEquals(new File(new File(git.getRepository().getDirectory(),
"modules"), walk.getPath()), subRepo.getDirectory());
}
File directory = createTempDirectory("testCloneRepositoryWithSubmodules");
CloneCommand clone = Git.cloneRepository();
clone.setDirectory(directory);
clone.setCloneSubmodules(true);
clone.setURI(fileUri());
Git git2 = clone.call();
addRepoToClose(git2.getRepository());
assertEquals(Constants.MASTER, git2.getRepository().getBranch());
assertTrue(new File(git2.getRepository().getWorkTree(), path
+ File.separatorChar + file).exists());
SubmoduleStatusCommand status = new SubmoduleStatusCommand(
git2.getRepository());
Map<String, SubmoduleStatus> statuses = status.call();
SubmoduleStatus pathStatus = statuses.get(path);
assertNotNull(pathStatus);
assertEquals(SubmoduleStatusType.INITIALIZED, pathStatus.getType());
assertEquals(commit, pathStatus.getHeadId());
assertEquals(commit, pathStatus.getIndexId());
try (SubmoduleWalk walk = SubmoduleWalk
.forIndex(git2.getRepository())) {
assertTrue(walk.next());
Repository clonedSub1 = walk.getRepository();
addRepoToClose(clonedSub1);
assertNotNull(clonedSub1);
assertEquals(new File(git2.getRepository().getWorkTree(),
walk.getPath()), clonedSub1.getWorkTree());
assertEquals(
new File(new File(git2.getRepository().getDirectory(),
"modules"), walk.getPath()),
clonedSub1.getDirectory());
}
}
@Test
public void testCloneRepositoryWithNestedSubmodules() throws Exception {
git.checkout().setName(Constants.MASTER).call();
// Create submodule 1
File submodule1 = createTempDirectory("testCloneRepositoryWithNestedSubmodules1");
Git sub1Git = Git.init().setDirectory(submodule1).call();
assertNotNull(sub1Git);
Repository sub1 = sub1Git.getRepository();
assertNotNull(sub1);
addRepoToClose(sub1);
String file = "file.txt";
String path = "sub";
write(new File(sub1.getWorkTree(), file), "content");
sub1Git.add().addFilepattern(file).call();
RevCommit commit = sub1Git.commit().setMessage("create file").call();
assertNotNull(commit);
// Create submodule 2
File submodule2 = createTempDirectory("testCloneRepositoryWithNestedSubmodules2");
Git sub2Git = Git.init().setDirectory(submodule2).call();
assertNotNull(sub2Git);
Repository sub2 = sub2Git.getRepository();
assertNotNull(sub2);
addRepoToClose(sub2);
write(new File(sub2.getWorkTree(), file), "content");
sub2Git.add().addFilepattern(file).call();
RevCommit sub2Head = sub2Git.commit().setMessage("create file").call();
assertNotNull(sub2Head);
// Add submodule 2 to submodule 1
Repository r = sub1Git.submoduleAdd().setPath(path)
.setURI(sub2.getDirectory().toURI().toString()).call();
assertNotNull(r);
addRepoToClose(r);
RevCommit sub1Head = sub1Git.commit().setAll(true)
.setMessage("Adding submodule").call();
assertNotNull(sub1Head);
// Add submodule 1 to default repository
r = git.submoduleAdd().setPath(path)
.setURI(sub1.getDirectory().toURI().toString()).call();
assertNotNull(r);
addRepoToClose(r);
assertNotNull(git.commit().setAll(true).setMessage("Adding submodule")
.call());
// Clone default repository and include submodules
File directory = createTempDirectory("testCloneRepositoryWithNestedSubmodules");
CloneCommand clone = Git.cloneRepository();
clone.setDirectory(directory);
clone.setCloneSubmodules(true);
clone.setURI(git.getRepository().getDirectory().toURI().toString());
Git git2 = clone.call();
addRepoToClose(git2.getRepository());
assertEquals(Constants.MASTER, git2.getRepository().getBranch());
assertTrue(new File(git2.getRepository().getWorkTree(), path
+ File.separatorChar + file).exists());
assertTrue(new File(git2.getRepository().getWorkTree(), path
+ File.separatorChar + path + File.separatorChar + file)
.exists());
SubmoduleStatusCommand status = new SubmoduleStatusCommand(
git2.getRepository());
Map<String, SubmoduleStatus> statuses = status.call();
SubmoduleStatus pathStatus = statuses.get(path);
assertNotNull(pathStatus);
assertEquals(SubmoduleStatusType.INITIALIZED, pathStatus.getType());
assertEquals(sub1Head, pathStatus.getHeadId());
assertEquals(sub1Head, pathStatus.getIndexId());
try (SubmoduleWalk walk = SubmoduleWalk
.forIndex(git2.getRepository())) {
assertTrue(walk.next());
try (Repository clonedSub1 = walk.getRepository()) {
assertNotNull(clonedSub1);
assertEquals(new File(git2.getRepository().getWorkTree(),
walk.getPath()), clonedSub1.getWorkTree());
assertEquals(
new File(new File(git2.getRepository().getDirectory(),
"modules"), walk.getPath()),
clonedSub1.getDirectory());
status = new SubmoduleStatusCommand(clonedSub1);
statuses = status.call();
}
assertFalse(walk.next());
}
pathStatus = statuses.get(path);
assertNotNull(pathStatus);
assertEquals(SubmoduleStatusType.INITIALIZED, pathStatus.getType());
assertEquals(sub2Head, pathStatus.getHeadId());
assertEquals(sub2Head, pathStatus.getIndexId());
}
@Test
public void testCloneWithAutoSetupRebase() throws Exception {
File directory = createTempDirectory("testCloneRepository1");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertNull(git2.getRepository().getConfig().getEnum(
BranchRebaseMode.values(),
ConfigConstants.CONFIG_BRANCH_SECTION, "test",
ConfigConstants.CONFIG_KEY_REBASE, null));
StoredConfig userConfig = SystemReader.getInstance()
.getUserConfig();
userConfig.setString(ConfigConstants.CONFIG_BRANCH_SECTION, null,
ConfigConstants.CONFIG_KEY_AUTOSETUPREBASE,
ConfigConstants.CONFIG_KEY_ALWAYS);
userConfig.save();
directory = createTempDirectory("testCloneRepository2");
command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals(BranchRebaseMode.REBASE,
git2.getRepository().getConfig().getEnum(
BranchRebaseMode.values(),
ConfigConstants.CONFIG_BRANCH_SECTION, "test",
ConfigConstants.CONFIG_KEY_REBASE,
BranchRebaseMode.NONE));
userConfig.setString(ConfigConstants.CONFIG_BRANCH_SECTION, null,
ConfigConstants.CONFIG_KEY_AUTOSETUPREBASE,
ConfigConstants.CONFIG_KEY_REMOTE);
userConfig.save();
directory = createTempDirectory("testCloneRepository2");
command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals(BranchRebaseMode.REBASE,
git2.getRepository().getConfig().getEnum(
BranchRebaseMode.values(),
ConfigConstants.CONFIG_BRANCH_SECTION, "test",
ConfigConstants.CONFIG_KEY_REBASE,
BranchRebaseMode.NONE));
}
@Test
public void testCloneWithPullMerge() throws Exception {
File directory = createTempDirectory("testCloneRepository1");
try (Git g = Git.init().setDirectory(directory).setBare(false).call()) {
g.remoteAdd().setName(Constants.DEFAULT_REMOTE_NAME)
.setUri(new URIish(fileUri())).call();
PullResult result = g.pull().setRebase(false).call();
assertTrue(result.isSuccessful());
assertEquals("refs/heads/master",
g.getRepository().getFullBranch());
checkFile(new File(directory, "Test.txt"), "Hello world");
}
}
@Test
public void testCloneWithPullRebase() throws Exception {
File directory = createTempDirectory("testCloneRepository1");
try (Git g = Git.init().setDirectory(directory).setBare(false).call()) {
g.remoteAdd().setName(Constants.DEFAULT_REMOTE_NAME)
.setUri(new URIish(fileUri())).call();
PullResult result = g.pull().setRebase(true).call();
assertTrue(result.isSuccessful());
assertEquals("refs/heads/master",
g.getRepository().getFullBranch());
checkFile(new File(directory, "Test.txt"), "Hello world");
}
}
@Test
public void testCloneNoTags() throws IOException, JGitInternalException,
GitAPIException, URISyntaxException {
File directory = createTempDirectory("testCloneRepository");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setNoTags();
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertNotNull(git2.getRepository().resolve("refs/heads/test"));
assertNull(git2.getRepository().resolve("tag-initial"));
assertNull(git2.getRepository().resolve("tag-for-blob"));
assertTagOption(git2.getRepository(), TagOpt.NO_TAGS);
}
@Test
public void testCloneFollowTags() throws IOException, JGitInternalException,
GitAPIException, URISyntaxException {
File directory = createTempDirectory("testCloneRepository");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setBranch("refs/heads/master");
command.setBranchesToClone(
Collections.singletonList("refs/heads/master"));
command.setTagOption(TagOpt.FETCH_TAGS);
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertNull(git2.getRepository().resolve("refs/heads/test"));
assertNotNull(git2.getRepository().resolve("tag-initial"));
assertNotNull(git2.getRepository().resolve("tag-for-blob"));
assertTagOption(git2.getRepository(), TagOpt.FETCH_TAGS);
}
@Test
public void testCloneWithHeadSymRefIsMasterCopy() throws IOException, GitAPIException {
// create a branch with the same head as master and switch to it
git.checkout().setStartPoint("master").setCreateBranch(true).setName("master-copy").call();
// when we clone the HEAD symref->master-copy means we start on master-copy and not master
File directory = createTempDirectory("testCloneRepositorySymRef_master-copy");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("refs/heads/master-copy", git2.getRepository().getFullBranch());
}
@Test
public void testCloneWithHeadSymRefIsNonMasterCopy() throws IOException, GitAPIException {
// create a branch with the same head as test and switch to it
git.checkout().setStartPoint("test").setCreateBranch(true).setName("test-copy").call();
File directory = createTempDirectory("testCloneRepositorySymRef_test-copy");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
Git git2 = command.call();
addRepoToClose(git2.getRepository());
assertEquals("refs/heads/test-copy", git2.getRepository().getFullBranch());
}
@Test
public void testCloneRepositoryWithDepth() throws IOException, JGitInternalException, GitAPIException {
File directory = createTempDirectory("testCloneRepositoryWithDepth");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setDepth(1);
command.setBranchesToClone(Set.of("refs/heads/test"));
Git git2 = command.call();
addRepoToClose(git2.getRepository());
List<RevCommit> log = StreamSupport.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(1, log.size());
RevCommit commit = log.get(0);
assertEquals(Set.of(commit.getId()),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals("Second commit", commit.getFullMessage());
assertEquals(0, commit.getParentCount());
}
@Test
public void testCloneRepositoryWithDepthAndAllBranches() throws IOException, JGitInternalException, GitAPIException {
File directory = createTempDirectory("testCloneRepositoryWithDepthAndAllBranches");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setDepth(1);
command.setCloneAllBranches(true);
Git git2 = command.call();
addRepoToClose(git2.getRepository());
List<RevCommit> log = StreamSupport.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(2, log.size());
assertEquals(log.stream().map(RevCommit::getId).collect(Collectors.toSet()),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals(List.of("Second commit", "Initial commit"),
log.stream().map(RevCommit::getFullMessage).collect(Collectors.toList()));
for (RevCommit commit : log) {
assertEquals(0, commit.getParentCount());
}
}
@Test
public void testCloneRepositoryWithDepth2() throws Exception {
RevCommit parent = tr.git().log().call().iterator().next();
RevCommit commit = tr.commit()
.parent(parent)
.message("Third commit")
.add("test.txt", "Hello world")
.create();
tr.update("refs/heads/test", commit);
File directory = createTempDirectory("testCloneRepositoryWithDepth2");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setDepth(2);
command.setBranchesToClone(Set.of("refs/heads/test"));
Git git2 = command.call();
addRepoToClose(git2.getRepository());
List<RevCommit> log = StreamSupport
.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(2, log.size());
assertEquals(Set.of(parent.getId()),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals(List.of("Third commit", "Second commit"), log.stream()
.map(RevCommit::getFullMessage).collect(Collectors.toList()));
assertEquals(List.of(Integer.valueOf(1), Integer.valueOf(0)),
log.stream().map(RevCommit::getParentCount)
.collect(Collectors.toList()));
}
@Test
public void testCloneRepositoryWithDepthAndFetch() throws Exception {
File directory = createTempDirectory("testCloneRepositoryWithDepthAndFetch");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setDepth(1);
command.setBranchesToClone(Set.of("refs/heads/test"));
Git git2 = command.call();
addRepoToClose(git2.getRepository());
RevCommit parent = tr.git().log().call().iterator().next();
RevCommit commit = tr.commit()
.parent(parent)
.message("Third commit")
.add("test.txt", "Hello world")
.create();
tr.update("refs/heads/test", commit);
git2.fetch().call();
List<RevCommit> log = StreamSupport
.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(2, log.size());
assertEquals(Set.of(parent.getId()),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals(List.of("Third commit", "Second commit"), log.stream()
.map(RevCommit::getFullMessage).collect(Collectors.toList()));
assertEquals(List.of(Integer.valueOf(1), Integer.valueOf(0)),
log.stream().map(RevCommit::getParentCount)
.collect(Collectors.toList()));
}
@Test
public void testCloneRepositoryWithDepthAndFetchWithDepth() throws Exception {
File directory = createTempDirectory("testCloneRepositoryWithDepthAndFetchWithDepth");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setDepth(1);
command.setBranchesToClone(Set.of("refs/heads/test"));
Git git2 = command.call();
addRepoToClose(git2.getRepository());
RevCommit parent = tr.git().log().call().iterator().next();
RevCommit commit = tr.commit()
.parent(parent)
.message("Third commit")
.add("test.txt", "Hello world")
.create();
tr.update("refs/heads/test", commit);
git2.fetch().setDepth(1).call();
List<RevCommit> log = StreamSupport
.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(2, log.size());
assertEquals(
log.stream().map(RevObject::getId).collect(Collectors.toSet()),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals(List.of("Third commit", "Second commit"), log.stream()
.map(RevCommit::getFullMessage).collect(Collectors.toList()));
assertEquals(List.of(Integer.valueOf(0), Integer.valueOf(0)),
log.stream().map(RevCommit::getParentCount)
.collect(Collectors.toList()));
}
@Test
public void testCloneRepositoryWithDepthAndFetchUnshallow() throws Exception {
File directory = createTempDirectory("testCloneRepositoryWithDepthAndFetchUnshallow");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setDepth(1);
command.setBranchesToClone(Set.of("refs/heads/test"));
Git git2 = command.call();
addRepoToClose(git2.getRepository());
git2.fetch().setUnshallow(true).call();
List<RevCommit> log = StreamSupport
.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(2, log.size());
assertEquals(Set.of(),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals(List.of("Second commit", "Initial commit"), log.stream()
.map(RevCommit::getFullMessage).collect(Collectors.toList()));
assertEquals(List.of(Integer.valueOf(1), Integer.valueOf(0)),
log.stream().map(RevCommit::getParentCount)
.collect(Collectors.toList()));
}
@Test
public void testCloneRepositoryWithShallowSince() throws Exception {
RevCommit commit = tr.commit()
.parent(tr.git().log().call().iterator().next())
.message("Third commit").add("test.txt", "Hello world")
.create();
tr.update("refs/heads/test", commit);
File directory = createTempDirectory("testCloneRepositoryWithShallowSince");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.setShallowSince(Instant.ofEpochSecond(commit.getCommitTime()));
command.setBranchesToClone(Set.of("refs/heads/test"));
Git git2 = command.call();
addRepoToClose(git2.getRepository());
List<RevCommit> log = StreamSupport
.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(1, log.size());
assertEquals(Set.of(commit.getId()),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals("Third commit", log.get(0).getFullMessage());
assertEquals(0, log.get(0).getParentCount());
}
@Test
public void testCloneRepositoryWithShallowExclude() throws Exception {
RevCommit parent = tr.git().log().call().iterator().next();
tr.update("refs/heads/test",
tr.commit()
.parent(parent)
.message("Third commit")
.add("test.txt", "Hello world")
.create());
File directory = createTempDirectory("testCloneRepositoryWithShallowExclude");
CloneCommand command = Git.cloneRepository();
command.setDirectory(directory);
command.setURI(fileUri());
command.addShallowExclude(parent.getId());
command.setBranchesToClone(Set.of("refs/heads/test"));
Git git2 = command.call();
addRepoToClose(git2.getRepository());
List<RevCommit> log = StreamSupport
.stream(git2.log().all().call().spliterator(), false)
.collect(Collectors.toList());
assertEquals(1, log.size());
RevCommit commit = log.get(0);
assertEquals(Set.of(commit.getId()),
git2.getRepository().getObjectDatabase().getShallowCommits());
assertEquals("Third commit", commit.getFullMessage());
assertEquals(0, commit.getParentCount());
}
private void assertTagOption(Repository repo, TagOpt expectedTagOption)
throws URISyntaxException {
RemoteConfig remoteConfig = new RemoteConfig(
repo.getConfig(), "origin");
assertEquals(expectedTagOption, remoteConfig.getTagOpt());
}
private String fileUri() {
return "file://" + git.getRepository().getWorkTree().getAbsolutePath();
}
}