Come “gatto” un file in JGit?
Domanda
Qualche tempo fa ero alla ricerca di un embeddable sistema di controllo di versione distribuito in Java , e penso che ho trovato in JGit , che è una pura implementazione Java di git. Tuttavia, non c'è molto in termini di codice di esempio o tutorial.
Come posso utilizzare JGit per recuperare la versione HEAD di un certo file (proprio come svn cat
o hg cat
whould fare)?
Suppongo che questo comporta qualche rev-tree-walking e sono alla ricerca di un esempio di codice.
Soluzione
Purtroppo la risposta di Thilo non funziona con l'ultima API JGit. Ecco la soluzione che ho trovato:
File repoDir = new File("test-git");
// open the repository
Repository repository = new Repository(repoDir);
// find the HEAD
ObjectId lastCommitId = repository.resolve(Constants.HEAD);
// now we have to get the commit
RevWalk revWalk = new RevWalk(repository);
RevCommit commit = revWalk.parseCommit(lastCommitId);
// and using commit's tree find the path
RevTree tree = commit.getTree();
TreeWalk treeWalk = new TreeWalk(repository);
treeWalk.addTree(tree);
treeWalk.setRecursive(true);
treeWalk.setFilter(PathFilter.create(path));
if (!treeWalk.next()) {
return null;
}
ObjectId objectId = treeWalk.getObjectId(0);
ObjectLoader loader = repository.open(objectId);
// and then one can use either
InputStream in = loader.openStream()
// or
loader.copyTo(out)
Vorrei che fosse più semplice.
Altri suggerimenti
Ecco una versione più semplice di @ di morisil risposta, utilizzando alcuni dei concetti da laugh @directed di e testato con JGit 2.2.0:
private String fetchBlob(String revSpec, String path) throws MissingObjectException, IncorrectObjectTypeException,
IOException {
// Resolve the revision specification
final ObjectId id = this.repo.resolve(revSpec);
// Makes it simpler to release the allocated resources in one go
ObjectReader reader = this.repo.newObjectReader();
try {
// Get the commit object for that revision
RevWalk walk = new RevWalk(reader);
RevCommit commit = walk.parseCommit(id);
// Get the revision's file tree
RevTree tree = commit.getTree();
// .. and narrow it down to the single file's path
TreeWalk treewalk = TreeWalk.forPath(reader, path, tree);
if (treewalk != null) {
// use the blob id to read the file's data
byte[] data = reader.open(treewalk.getObjectId(0)).getBytes();
return new String(data, "utf-8");
} else {
return "";
}
} finally {
reader.release();
}
}
repo
è un oggetto repository come creata nelle altre risposte.
Ho seguito @ Thilo e di risposta di @ morisil per ottenere questo, compatibile con JGit 1.2.0:
File repoDir = new File("test-git/.git");
// open the repository
Repository repo = new Repository(repoDir);
// find the HEAD
Commit head = repo.mapCommit(Constants.HEAD);
// retrieve the tree in HEAD
Tree tree = head.getTree();
// 1.2.0 api version here
// find a file (as a TreeEntry, which contains the blob object id)
TreeWalk treewalk = TreeWalk.forPath(repo, "b/test.txt", tree);
// use the blob id to read the file's data
byte[] data = repo.open(treewalk.getObjectId(0)).getBytes();
non ho la prova della versione di Java ma dovrebbe funzionare. Traduce dal
(.getBytes (.open repo (.getObjectId (TreeWalk/forPath repo "b/test.txt" tree) 0)))
in clojure (seguendo la stessa configurazione come sezione superiore), che funziona.
capito da solo. L'API è piuttosto basso livello, ma non è troppo male:
File repoDir = new File("test-git/.git");
// open the repository
Repository repo = new Repository(repoDir);
// find the HEAD
Commit head = repo.mapCommit(Constants.HEAD);
// retrieve the tree in HEAD
Tree tree = head.getTree();
// find a file (as a TreeEntry, which contains the blob object id)
TreeEntry entry = tree.findBlobMember("b/test.txt");
// use the blob id to read the file's data
byte[] data = repo.openBlob(entry.getId()).getBytes();
Ho iniziato a scrivere una libreria chiamata gitective che contiene molti aiutanti per lavorare con blob, commette, e alberi che utilizzano JGit ed è MIT-licenza ed è disponibile su GitHub.
contenuto del file in HEAD commettere
Repository repo = new FileRepository("/repos/project/.git");
String content = BlobUtils.getHeadContent(repo, "src/Buffer.java");
il contenuto di un file su un ramo
Repository repo = new FileRepository("/repos/project/.git");
String content = BlobUtils.getContent(repo, "master", "src/Buffer.java");
Diff due file
Repository repo = new FileRepository("/repos/project/.git");
ObjectId current = BlobUtils.getId(repo, "master", "Main.java");
ObjectId previous = BlobUtils.getId(repo, "master~1", "Main.java");
Collection<Edit> edit = BlobUtils.diff(repo, previous, current);
Altri esempi di utilità fornite sono dettagliati nella README .
Ci sono alcune informazioni a JGit Tutorial (ma che anche non è né veramente utile né completa e probabilmente superati da quando sono passati a eclisse dove nessuna documentazione è ancora disponibile).
È possibile leggere il contenuto di un determinato percorso file come segue. Si prega di essere consapevole del fatto che il TreeWalk può essere nullo se nessun percorso è stato trovato nella struttura di data. Quindi richiede alcuni trattamenti specifici.
public String readFile(RevCommit commit, String filepath) throws IOException {
try (TreeWalk walk = TreeWalk.forPath(repo, filepath, commit.getTree())) {
if (walk != null) {
byte[] bytes = repo.open(walk.getObjectId(0)).getBytes();
return new String(bytes, StandardCharsets.UTF_8);
} else {
throw new IllegalArgumentException("No path found.");
}
}
}
Ad esempio:
ObjectId head = repo.resolve(Constants.HEAD);
RevCommit last = repo.parseCommit(head);
readFile(last, "docs/README.md")
Questa risposta è scritta con JGit 4.8.0.