diff options
author | James Moger <james.moger@gitblit.com> | 2011-06-07 19:08:24 -0400 |
---|---|---|
committer | James Moger <james.moger@gitblit.com> | 2011-06-07 19:08:24 -0400 |
commit | 16856603ec575718857768e2d18e455c95fd6ea4 (patch) | |
tree | 7369326e7de9dec82d80d63a671b54f3d69f8210 | |
parent | f08c1ca55e1ffaef81f3d6514aa4bffa5d716c5b (diff) | |
download | gitblit-16856603ec575718857768e2d18e455c95fd6ea4.tar.gz gitblit-16856603ec575718857768e2d18e455c95fd6ea4.zip |
Documentation. Moved clone and fetch into JGitUtils. Create bare only.
-rw-r--r-- | docs/00_index.mkd | 2 | ||||
-rw-r--r-- | docs/00_setup.mkd | 8 | ||||
-rw-r--r-- | docs/01_faq.mkd | 23 | ||||
-rw-r--r-- | docs/architecture.odg | bin | 15758 -> 15754 bytes | |||
-rw-r--r-- | docs/architecture.png | bin | 28259 -> 71104 bytes | |||
-rw-r--r-- | src/com/gitblit/GitBlit.java | 4 | ||||
-rw-r--r-- | src/com/gitblit/utils/JGitUtils.java | 58 | ||||
-rw-r--r-- | tests/com/gitblit/tests/GitBlitSuite.java | 64 | ||||
-rw-r--r-- | tests/com/gitblit/tests/JGitUtilsTest.java | 22 |
9 files changed, 110 insertions, 71 deletions
diff --git a/docs/00_index.mkd b/docs/00_index.mkd index 09027339..8c8f9322 100644 --- a/docs/00_index.mkd +++ b/docs/00_index.mkd @@ -31,6 +31,7 @@ sources @ [Github][gitbltsrc] - Repository Owners may edit repositories through the web UI
- Automatically generates a self-signed certificate for https communications
- Git-notes support
+- Branch-selectable metrics
- Dates can optionally be displayed using the browser's reported timezone
- Author and Committer email address display can be controlled
- Dynamic zip downloads feature
@@ -57,6 +58,7 @@ sources @ [Github][gitbltsrc] - Unit testing
- Branch selector on Metrics
- Blame
+- Clone remote repository
### Idea List
- Ticgit activity/timeline
diff --git a/docs/00_setup.mkd b/docs/00_setup.mkd index 48e02b7a..fc2fd852 100644 --- a/docs/00_setup.mkd +++ b/docs/00_setup.mkd @@ -9,7 +9,7 @@ Open `gitblit.properties` in your favorite text editor and make sure to review a - *server.httpBindInterface* and *server.httpsBindInterface*<br/>
**NOTE:** Consider using **https** exclusively because passwords for authentication are transmitted as clear text!
- *server.storePassword*<br/>
-**NOTE:** The certificate password AND the keystore password must match!
+**NOTE:** If you manually generate an ssl certificate, the certificate password AND the keystore password must match!
3. Execute `gitblit.cmd` or `java -jar gitblit.jar` from a command-line
4. Wait a minute or two while all dependencies are downloaded and your self-signed certificate is generated.
5. Open your browser to <http://localhost> or <https://localhost> depending on your chosen configuration.
@@ -35,9 +35,9 @@ All repository settings are stored within the repository `.git/config` file unde Repository names must be unique and are CASE-SENSITIVE ON CASE-SENSITIVE FILESYSTEMS. The name must be composed of letters, digits, or `/ _ - .`<br/>
Whitespace is illegal.
-Repositories can be grouped by folders. e.g. *libraries/mycoollib.git* and *libraries/myotherlib.git*
+Repositories can be grouped within subfolders. e.g. *libraries/mycoollib.git* and *libraries/myotherlib.git*
-Repository names will automatically have *.git* appended to the name at creation time, if not already specified.
+All created repositories are *bare* and will automatically have *.git* appended to the name at creation time, if not already specified.
#### Repository Owner
The *Repository Owner* has the special permission of being able to edit a repository through the web UI. The Repository Owner is not permitted to rename the repository, delete the repository, or reassign ownership to another user.
@@ -61,7 +61,7 @@ There is only one actual *role* in Gitblit and that is *#admin* which grants adm ### Creating your own Self-Signed Certificate
Review the contents of the `makekeystore.cmd` or `makekeystore_jdk.cmd` script and execute it.<br/>
-**NOTE:** The certificate password AND the keystore password must match!
+**NOTE:** If you manually generate an ssl certificate, the certificate password AND the keystore password must match!
### Running as a Service
Review the contents of the `installService.cmd` or `installService64.cmd`, as appropriate for your installed Java Virtual Machine.<br/>
diff --git a/docs/01_faq.mkd b/docs/01_faq.mkd index 5068e79a..a80e4e29 100644 --- a/docs/01_faq.mkd +++ b/docs/01_faq.mkd @@ -27,10 +27,30 @@ It's a phonetic play on [bitblt][bitblt] which is an image processing operation ### Why use Gitblit?
It's a small tool that allows you to easily manage shared repositories and doesn't require alot of setup or git kung-foo.
+### Who is the target user for Gitblit?
+Small workgroups that require centralized repositories.
+
+Gitblit is not meant to be a social coding resource like [Github](http://github.com) or [Bitbucket](http://bitbucket.com) with 100s or 1000s of users. Gitblit is designed to fulfill the same function as your centralized Subversion or CVS server.
+
+### Why does Gitblit exist?
+As a Java developer I prefer that as much of my tooling as possible is Java.<br/>
+Originally, I was going to use [Mercurial](http://mercurial.selenic.com) but...
+
+- MercurialEclipse [shells to Python and captures System.in](http://mercurial.808500.n3.nabble.com/Hg4J-Mercurial-pure-Java-library-tp2693090p2694555.html)<br/>
+Parsing command-line output is fragile and suboptimal.<br/>Unfortunately this is necessary because Mercurial is an application, not a library.
+- Mercurial seems to [frown](http://mercurial.808500.n3.nabble.com/Hg4J-Mercurial-pure-Java-library-tp2693090p2695051.html) on the fledgling [Hg4j][hg4j] (pure Java Mercurial) project.
+- Mercurial HTTP/HTTPS needs to run as CGI through Apache/IIS/etc, as mod_python through Apache, or served with a built-in http server.<br/>
+This requires setup and maintenance of multiple, mixed 3rd party components.
+
+Gitblit eliminates all that complication with its 100% Java stack and simple single configuration file.
+
### Do I need real Git?
No. Gitblit is based on [JGit][jgit] which is a pure Java implementation of the [Git version control system][git].<br/>
Everything you need for Gitblit is either in the zip distribution file or automatically downloaded on execution.
+### Can I run Gitblit in conjunction with my existing Git tooling?
+Yes. You can configure Gitblit to only be a repository viewer.
+
### Do I need a JDK or can I use a JRE?
Gitblit will run just fine with a JRE. Gitblit can optionally use `keytool` from the JDK to generate self-signed certificates, but normally Gitblit uses [BouncyCastle][bouncycastle] for that need.
@@ -73,4 +93,5 @@ Currently blame is not implemented. Those links are placeholders to remind me w [jgit]: http://eclipse.org/jgit "Eclipse JGit Site"
[git]: http://git-scm.com "Official Git Site"
[mina]: http://mina.apache.org "Apache Mina"
-[bouncycastle]: http://bouncycastle.org "The Legion of the Bouncy Castle"
\ No newline at end of file +[bouncycastle]: http://bouncycastle.org "The Legion of the Bouncy Castle"
+[hg4j]: http://code.google.com/p/hg4j/ "hg4j"
\ No newline at end of file diff --git a/docs/architecture.odg b/docs/architecture.odg Binary files differindex c2fc25c0..fd35f655 100644 --- a/docs/architecture.odg +++ b/docs/architecture.odg diff --git a/docs/architecture.png b/docs/architecture.png Binary files differindex 0dd7ddc2..c61881c3 100644 --- a/docs/architecture.png +++ b/docs/architecture.png diff --git a/src/com/gitblit/GitBlit.java b/src/com/gitblit/GitBlit.java index 595a5ee1..dcf5a6b2 100644 --- a/src/com/gitblit/GitBlit.java +++ b/src/com/gitblit/GitBlit.java @@ -226,7 +226,7 @@ public class GitBlit implements ServletContextListener { Repository r = null;
if (isCreate) {
// ensure created repository name ends with .git
- if (!repository.name.endsWith(org.eclipse.jgit.lib.Constants.DOT_GIT_EXT)) {
+ if (!repository.name.toLowerCase().endsWith(org.eclipse.jgit.lib.Constants.DOT_GIT_EXT)) {
repository.name += org.eclipse.jgit.lib.Constants.DOT_GIT_EXT;
}
if (new File(repositoriesFolder, repository.name).exists()) {
@@ -236,7 +236,7 @@ public class GitBlit implements ServletContextListener { }
// create repository
logger.info("create repository " + repository.name);
- r = JGitUtils.createRepository(repositoriesFolder, repository.name, true);
+ r = JGitUtils.createRepository(repositoriesFolder, repository.name);
} else {
// rename repository
if (!repositoryName.equalsIgnoreCase(repository.name)) {
diff --git a/src/com/gitblit/utils/JGitUtils.java b/src/com/gitblit/utils/JGitUtils.java index 6e02b9c4..f6d7108b 100644 --- a/src/com/gitblit/utils/JGitUtils.java +++ b/src/com/gitblit/utils/JGitUtils.java @@ -22,6 +22,7 @@ import java.io.InputStream; import java.io.OutputStream;
import java.nio.charset.Charset;
import java.util.ArrayList;
+import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
@@ -32,6 +33,8 @@ import java.util.Map.Entry; import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;
+import org.eclipse.jgit.api.CloneCommand;
+import org.eclipse.jgit.api.FetchCommand;
import org.eclipse.jgit.api.Git;
import org.eclipse.jgit.diff.DiffEntry;
import org.eclipse.jgit.diff.DiffEntry.ChangeType;
@@ -57,6 +60,9 @@ import org.eclipse.jgit.revwalk.RevSort; import org.eclipse.jgit.revwalk.RevTree;
import org.eclipse.jgit.revwalk.RevWalk;
import org.eclipse.jgit.revwalk.filter.RevFilter;
+import org.eclipse.jgit.storage.file.FileRepository;
+import org.eclipse.jgit.transport.FetchResult;
+import org.eclipse.jgit.transport.RefSpec;
import org.eclipse.jgit.treewalk.TreeWalk;
import org.eclipse.jgit.treewalk.filter.AndTreeFilter;
import org.eclipse.jgit.treewalk.filter.OrTreeFilter;
@@ -90,8 +96,54 @@ public class JGitUtils { return r.toString().trim();
}
- public static Repository createRepository(File repositoriesFolder, String name, boolean bare) {
- Git git = Git.init().setDirectory(new File(repositoriesFolder, name)).setBare(bare).call();
+ public static FetchResult cloneRepository(File repositoriesFolder, String name, String fromUrl) throws Exception {
+ FetchResult result = null;
+ if (!name.toLowerCase().endsWith(Constants.DOT_GIT_EXT)) {
+ name += Constants.DOT_GIT_EXT;
+ }
+ File folder = new File(repositoriesFolder, name);
+ if (folder.exists()) {
+ File gitDir = FileKey.resolve(new File(repositoriesFolder, name), FS.DETECTED);
+ FileRepository repository = new FileRepository(gitDir);
+ result = fetchRepository(repository);
+ repository.close();
+ } else {
+ CloneCommand clone = new CloneCommand();
+ clone.setBare(true);
+ clone.setCloneAllBranches(true);
+ clone.setURI(fromUrl);
+ clone.setDirectory(folder);
+ clone.call();
+ // Now we have to fetch because CloneCommand doesn't fetch
+ // refs/notes nor does it allow manual RefSpec.
+ File gitDir = FileKey.resolve(new File(repositoriesFolder, name), FS.DETECTED);
+ FileRepository repository = new FileRepository(gitDir);
+ result = fetchRepository(repository);
+ repository.close();
+ }
+ return result;
+ }
+
+ public static FetchResult fetchRepository(Repository repository, RefSpec... refSpecs)
+ throws Exception {
+ Git git = new Git(repository);
+ FetchCommand fetch = git.fetch();
+ List<RefSpec> specs = new ArrayList<RefSpec>();
+ if (refSpecs == null || refSpecs.length == 0) {
+ specs.add(new RefSpec("+refs/heads/*:refs/remotes/origin/*"));
+ specs.add(new RefSpec("+refs/tags/*:refs/tags/*"));
+ specs.add(new RefSpec("+refs/notes/*:refs/notes/*"));
+ } else {
+ specs.addAll(Arrays.asList(refSpecs));
+ }
+ fetch.setRefSpecs(specs);
+ FetchResult result = fetch.call();
+ repository.close();
+ return result;
+ }
+
+ public static Repository createRepository(File repositoriesFolder, String name) {
+ Git git = Git.init().setDirectory(new File(repositoriesFolder, name)).setBare(true).call();
return git.getRepository();
}
@@ -219,7 +271,7 @@ public class JGitUtils { refs.put(objectid, new ArrayList<RefModel>());
}
refs.get(objectid).add(ref);
- }
+ }
return refs;
}
diff --git a/tests/com/gitblit/tests/GitBlitSuite.java b/tests/com/gitblit/tests/GitBlitSuite.java index fe201b8b..e13e1bbf 100644 --- a/tests/com/gitblit/tests/GitBlitSuite.java +++ b/tests/com/gitblit/tests/GitBlitSuite.java @@ -16,25 +16,20 @@ package com.gitblit.tests;
import java.io.File;
-import java.util.ArrayList;
-import java.util.List;
import junit.extensions.TestSetup;
import junit.framework.Test;
import junit.framework.TestSuite;
-import org.eclipse.jgit.api.CloneCommand;
-import org.eclipse.jgit.api.FetchCommand;
-import org.eclipse.jgit.api.Git;
import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.storage.file.FileRepository;
-import org.eclipse.jgit.transport.RefSpec;
import com.gitblit.FileSettings;
import com.gitblit.GitBlit;
import com.gitblit.GitBlitException;
import com.gitblit.JettyLoginService;
import com.gitblit.models.RepositoryModel;
+import com.gitblit.utils.JGitUtils;
public class GitBlitSuite extends TestSetup {
public static final File REPOSITORIES = new File("git");
@@ -66,11 +61,11 @@ public class GitBlitSuite extends TestSetup { }
public static Repository getJGitRepository() throws Exception {
- return new FileRepository(new File(REPOSITORIES, "nested/jgit.git"));
+ return new FileRepository(new File(REPOSITORIES, "test/jgit.git"));
}
public static Repository getBluezGnomeRepository() throws Exception {
- return new FileRepository(new File(REPOSITORIES, "nested/bluez-gnome.git"));
+ return new FileRepository(new File(REPOSITORIES, "test/bluez-gnome.git"));
}
@Override
@@ -82,51 +77,24 @@ public class GitBlitSuite extends TestSetup { GitBlit.self().setLoginService(loginService);
if (REPOSITORIES.exists() || REPOSITORIES.mkdirs()) {
- cloneOrFetch("helloworld.git", "https://github.com/git/hello-world.git", true);
- cloneOrFetch("ticgit.git", "https://github.com/jeffWelling/ticgit.git", true);
- cloneOrFetch("nested/bluez-gnome.git", "https://git.kernel.org/pub/scm/bluetooth/bluez-gnome.git", true);
- cloneOrFetch("nested/jgit.git", "https://github.com/eclipse/jgit.git", true);
- cloneOrFetch("nested/helloworld.git", "https://github.com/git/hello-world.git", true);
+ cloneOrFetch("helloworld.git", "https://github.com/git/hello-world.git");
+ cloneOrFetch("ticgit.git", "https://github.com/jeffWelling/ticgit.git");
+ cloneOrFetch("test/bluez-gnome.git",
+ "https://git.kernel.org/pub/scm/bluetooth/bluez-gnome.git");
+ cloneOrFetch("test/jgit.git", "https://github.com/eclipse/jgit.git");
+ cloneOrFetch("test/helloworld.git", "https://github.com/git/hello-world.git");
enableTickets("ticgit.git");
enableDocs("ticgit.git");
showRemoteBranches("ticgit.git");
- showRemoteBranches("nested/jgit.git");
+ showRemoteBranches("test/jgit.git");
}
}
- private void cloneOrFetch(String toFolder, String fromUrl, boolean bare) throws Exception {
- File folder = new File(REPOSITORIES, toFolder + (bare ? "" : "/.git"));
- if (folder.exists()) {
- System.out.print("Updating " + (bare ? "bare " : " ") + toFolder + "... ");
- fetch(toFolder);
- System.out.println("done.");
- } else {
- System.out.println("Cloning " + (bare ? "bare " : " ") + toFolder + "... ");
- CloneCommand clone = new CloneCommand();
- clone.setBare(bare);
- clone.setCloneAllBranches(true);
- clone.setURI(fromUrl);
- clone.setDirectory(folder);
- clone.call();
- // Now we have to fetch because CloneCommand doesn't fetch
- // Notes nor does it allow manual RefSpec.
- fetch(toFolder);
- System.out.println("done.");
- }
- }
-
- private void fetch(String toFolder) throws Exception {
- FileRepository repository = new FileRepository(new File(REPOSITORIES, toFolder));
- Git git = new Git(repository);
- FetchCommand fetch = git.fetch();
- List<RefSpec> specs = new ArrayList<RefSpec>();
- specs.add(new RefSpec("+refs/heads/*:refs/remotes/origin/*"));
- specs.add(new RefSpec("+refs/tags/*:refs/tags/*"));
- specs.add(new RefSpec("+refs/notes/*:refs/notes/*"));
- fetch.setRefSpecs(specs);
- fetch.call();
- repository.close();
+ private void cloneOrFetch(String name, String fromUrl) throws Exception {
+ System.out.print("Fetching " + name + "... ");
+ JGitUtils.cloneRepository(REPOSITORIES, name, fromUrl);
+ System.out.println("done.");
}
private void enableTickets(String repositoryName) {
@@ -138,7 +106,7 @@ public class GitBlitSuite extends TestSetup { g.printStackTrace();
}
}
-
+
private void enableDocs(String repositoryName) {
try {
RepositoryModel model = GitBlit.self().getRepositoryModel(repositoryName);
@@ -148,7 +116,7 @@ public class GitBlitSuite extends TestSetup { g.printStackTrace();
}
}
-
+
private void showRemoteBranches(String repositoryName) {
try {
RepositoryModel model = GitBlit.self().getRepositoryModel(repositoryName);
diff --git a/tests/com/gitblit/tests/JGitUtilsTest.java b/tests/com/gitblit/tests/JGitUtilsTest.java index 6646bf80..277166cf 100644 --- a/tests/com/gitblit/tests/JGitUtilsTest.java +++ b/tests/com/gitblit/tests/JGitUtilsTest.java @@ -30,7 +30,9 @@ import org.eclipse.jgit.lib.FileMode; import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.PersonIdent;
import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.lib.RepositoryCache.FileKey;
import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.util.FS;
import com.gitblit.GitBlit;
import com.gitblit.Keys;
@@ -94,16 +96,10 @@ public class JGitUtilsTest extends TestCase { public void testCreateRepository() throws Exception {
String[] repositories = { "NewTestRepository.git", "NewTestRepository" };
- for (String repositoryName : repositories) {
- boolean isBare = repositoryName.endsWith(".git");
+ for (String repositoryName : repositories) {
Repository repository = JGitUtils.createRepository(GitBlitSuite.REPOSITORIES,
- repositoryName, isBare);
- File folder;
- if (isBare) {
- folder = new File(GitBlitSuite.REPOSITORIES, repositoryName);
- } else {
- folder = new File(GitBlitSuite.REPOSITORIES, repositoryName + "/.git");
- }
+ repositoryName);
+ File folder = FileKey.resolve(new File(GitBlitSuite.REPOSITORIES, repositoryName), FS.DETECTED);
assertTrue(repository != null);
assertFalse(JGitUtils.hasCommits(repository));
assertTrue(JGitUtils.getFirstCommit(repository, null) == null);
@@ -138,7 +134,7 @@ public class JGitUtilsTest extends TestCase { }
public void testBranches() throws Exception {
- Repository repository = GitBlitSuite.getTicgitRepository();
+ Repository repository = GitBlitSuite.getJGitRepository();
for (RefModel model : JGitUtils.getLocalBranches(repository, true, -1)) {
assertTrue(model.getName().startsWith(Constants.R_HEADS));
assertTrue(model.equals(model));
@@ -155,14 +151,14 @@ public class JGitUtilsTest extends TestCase { + model.getName().hashCode());
assertTrue(model.getShortMessage().equals(model.getShortMessage()));
}
- assertTrue(JGitUtils.getRemoteBranches(repository, true, 10).size() == 10);
+ assertTrue(JGitUtils.getRemoteBranches(repository, true, 8).size() == 8);
repository.close();
}
public void testTags() throws Exception {
- Repository repository = GitBlitSuite.getTicgitRepository();
+ Repository repository = GitBlitSuite.getJGitRepository();
for (RefModel model : JGitUtils.getTags(repository, true, -1)) {
- if (model.getObjectId().getName().equals("283035e4848054ff1803cb0e690270787dc92399")) {
+ if (model.getObjectId().getName().equals("d28091fb2977077471138fe97da1440e0e8ae0da")) {
assertTrue("Not an annotated tag!", model.isAnnotatedTag());
}
assertTrue(model.getName().startsWith(Constants.R_TAGS));
|