package service import java.io.File import java.util.Date import org.eclipse.jgit.api.Git import org.apache.commons.io.FileUtils import util.{StringUtil, Directory, JGitUtil, LockUtil} import _root_.util.ControlUtil._ import org.eclipse.jgit.treewalk.{TreeWalk, CanonicalTreeParser} import org.eclipse.jgit.diff.DiffFormatter import org.eclipse.jgit.api.errors.PatchApplyException import java.util import org.eclipse.jgit.lib._ import org.eclipse.jgit.dircache.{DirCache, DirCacheEntry} import org.eclipse.jgit.merge.{ResolveMerger, MergeStrategy} import scala.Some import org.eclipse.jgit.revwalk.RevWalk import org.eclipse.jgit.api.CheckoutCommand.Stage object WikiService { /** * The model for wiki page. * * @param name the page name * @param content the page content * @param committer the last committer * @param time the last modified time * @param id the latest commit id */ case class WikiPageInfo(name: String, content: String, committer: String, time: Date, id: String) /** * The model for wiki page history. * * @param name the page name * @param committer the committer the committer * @param message the commit message * @param date the commit date */ case class WikiPageHistoryInfo(name: String, committer: String, message: String, date: Date) } trait WikiService { import WikiService._ def createWikiRepository(loginAccount: model.Account, owner: String, repository: String): Unit = LockUtil.lock(s"${owner}/${repository}/wiki"){ defining(Directory.getWikiRepositoryDir(owner, repository)){ dir => if(!dir.exists){ try { JGitUtil.initRepository(dir) saveWikiPage(owner, repository, "Home", "Home", s"Welcome to the ${repository} wiki!!", loginAccount, "Initial Commit", None) } finally { // once delete cloned repository because initial cloned repository does not have 'branch.master.merge' FileUtils.deleteDirectory(Directory.getWikiWorkDir(owner, repository)) } } } } /** * Returns the wiki page. */ def getWikiPage(owner: String, repository: String, pageName: String): Option[WikiPageInfo] = { using(Git.open(Directory.getWikiRepositoryDir(owner, repository))){ git => optionIf(!JGitUtil.isEmpty(git)){ JGitUtil.getFileList(git, "master", ".").find(_.name == pageName + ".md").map { file => WikiPageInfo(file.name, new String(git.getRepository.open(file.id).getBytes, "UTF-8"), file.committer, file.time, file.commitId) } } } } /** * Returns the content of the specified file. */ def getFileContent(owner: String, repository: String, path: String): Option[Array[Byte]] = using(Git.open(Directory.getWikiRepositoryDir(owner, repository))){ git => optionIf(!JGitUtil.isEmpty(git)){ val index = path.lastIndexOf('/') val parentPath = if(index < 0) "." else path.substring(0, index) val fileName = if(index < 0) path else path.substring(index + 1) JGitUtil.getFileList(git, "master", parentPath).find(_.name == fileName).map { file => git.getRepository.open(file.id).getBytes } } } /** * Returns the list of wiki page names. */ def getWikiPageList(owner: String, repository: String): List[String] = { using(Git.open(Directory.getWikiRepositoryDir(owner, repository))){ git => JGitUtil.getFileList(git, "master", ".") .filter(_.name.endsWith(".md")) .map(_.name.replaceFirst("\\.md$", "")) .sortBy(x => x) } } /** * Reverts specified changes. */ def revertWikiPage(owner: String, repository: String, from: String, to: String, committer: model.Account, pageName: Option[String]): Boolean = { // LockUtil.lock(s"${owner}/${repository}/wiki"){ // defining(Directory.getWikiWorkDir(owner, repository)){ workDir => // // clone working copy // cloneOrPullWorkingCopy(workDir, owner, repository) // // using(Git.open(workDir)){ git => // val reader = git.getRepository.newObjectReader // val oldTreeIter = new CanonicalTreeParser // oldTreeIter.reset(reader, git.getRepository.resolve(from + "^{tree}")) // // val newTreeIter = new CanonicalTreeParser // newTreeIter.reset(reader, git.getRepository.resolve(to + "^{tree}")) // // import scala.collection.JavaConverters._ // val diffs = git.diff.setNewTree(oldTreeIter).setOldTree(newTreeIter).call.asScala.filter { diff => // pageName match { // case Some(x) => diff.getNewPath == x + ".md" // case None => true // } // } // // val patch = using(new java.io.ByteArrayOutputStream()){ out => // val formatter = new DiffFormatter(out) // formatter.setRepository(git.getRepository) // formatter.format(diffs.asJava) // new String(out.toByteArray, "UTF-8") // } // // try { // git.apply.setPatch(new java.io.ByteArrayInputStream(patch.getBytes("UTF-8"))).call // git.add.addFilepattern(".").call // git.commit.setCommitter(committer.fullName, committer.mailAddress).setMessage(pageName match { // case Some(x) => s"Revert ${from} ... ${to} on ${x}" // case None => s"Revert ${from} ... ${to}" // }).call // git.push.call // true // } catch { // case ex: PatchApplyException => false // } // } // } // } true } /** * Save the wiki page. */ def saveWikiPage(owner: String, repository: String, currentPageName: String, newPageName: String, content: String, committer: model.Account, message: String, currentId: Option[String]): Option[String] = { import scala.collection.JavaConverters._ // TODO conflict detection and rename page LockUtil.lock(s"${owner}/${repository}/wiki"){ using(Git.open(Directory.getWikiRepositoryDir(owner, repository))){ git => val repo = git.getRepository() val dirCache = DirCache.newInCore() val builder = dirCache.builder() val inserter = repo.newObjectInserter() val headId = repo.resolve(Constants.HEAD + "^{commit}") var created = true using(new RevWalk(git.getRepository)){ revWalk => val treeWalk = new TreeWalk(repo) val hIdx = treeWalk.addTree(revWalk.parseTree(headId)) treeWalk.setRecursive(true) while(treeWalk.next){ val path = treeWalk.getPathString if(path != newPageName + ".md"){ val hTree = treeWalk.getTree(hIdx, classOf[CanonicalTreeParser]) val entry = new DirCacheEntry(path) entry.setObjectId(hTree.getEntryObjectId()) entry.setFileMode(hTree.getEntryFileMode()) builder.add(entry) } else { created = false } } treeWalk.release() } val entry = new DirCacheEntry(newPageName + ".md") entry.setFileMode(FileMode.REGULAR_FILE) entry.setObjectId(inserter.insert(Constants.OBJ_BLOB, content.getBytes("UTF-8"))) builder.add(entry) builder.finish() val treeId = dirCache.writeTree(inserter) val newCommit = new CommitBuilder() newCommit.setCommitter(new PersonIdent(committer.fullName, committer.mailAddress)) newCommit.setAuthor(new PersonIdent(committer.fullName, committer.mailAddress)) newCommit.setMessage(if(message.trim.length == 0) { if(created){ s"Created ${newPageName}" } else { s"Updated ${newPageName}" } } else { message }) newCommit.setParentIds(List(headId).asJava) newCommit.setTreeId(treeId) val newHeadId = inserter.insert(newCommit) inserter.flush() val ru = repo.updateRef(Constants.HEAD) ru.setNewObjectId(newHeadId) ru.update() Some(newHeadId.getName) } } // defining(Directory.getWikiWorkDir(owner, repository)){ workDir => // // clone working copy // cloneOrPullWorkingCopy(workDir, owner, repository) // // // write as file // using(Git.open(workDir)){ git => // defining(new File(workDir, newPageName + ".md")){ file => // // new page // val created = !file.exists // // // created or updated // val added = executeIf(!file.exists || FileUtils.readFileToString(file, "UTF-8") != content){ // FileUtils.writeStringToFile(file, content, "UTF-8") // git.add.addFilepattern(file.getName).call // } // // // delete file // val deleted = executeIf(currentPageName != "" && currentPageName != newPageName){ // git.rm.addFilepattern(currentPageName + ".md").call // } // // // commit and push // optionIf(added || deleted){ // defining(git.commit.setCommitter(committer.fullName, committer.mailAddress) // .setMessage(if(message.trim.length == 0){ // if(deleted){ // s"Rename ${currentPageName} to ${newPageName}" // } else if(created){ // s"Created ${newPageName}" // } else { // s"Updated ${newPageName}" // } // } else { // message // }).call){ commit => // git.push.call // Some(commit.getName) // } // } // } // } // } // } } /** * Delete the wiki page. */ def deleteWikiPage(owner: String, repository: String, pageName: String, committer: String, mailAddress: String, message: String): Unit = { // LockUtil.lock(s"${owner}/${repository}/wiki"){ // defining(Directory.getWikiWorkDir(owner, repository)){ workDir => // // clone working copy // cloneOrPullWorkingCopy(workDir, owner, repository) // // // delete file // new File(workDir, pageName + ".md").delete // // using(Git.open(workDir)){ git => // git.rm.addFilepattern(pageName + ".md").call // // // commit and push // git.commit.setCommitter(committer, mailAddress).setMessage(message).call // git.push.call // } // } // } } // private def cloneOrPullWorkingCopy(workDir: File, owner: String, repository: String): Unit = { // if(!workDir.exists){ // Git.cloneRepository // .setURI(Directory.getWikiRepositoryDir(owner, repository).toURI.toString) // .setDirectory(workDir) // .call // .getRepository // .close // } else using(Git.open(workDir)){ git => // git.pull.call // } // } }