mirror of
https://github.com/gitbucket/gitbucket.git
synced 2025-11-08 22:45:51 +01:00
Merge branch 'master' of https://github.com/takezoe/gitbucket.git
This commit is contained in:
@@ -76,7 +76,7 @@ trait AccountControllerBase extends AccountManagementControllerBase with FlashMa
|
|||||||
|
|
||||||
updateImage(userName, form.fileId, form.clearImage)
|
updateImage(userName, form.fileId, form.clearImage)
|
||||||
flash += "info" -> "Account information has been updated."
|
flash += "info" -> "Account information has been updated."
|
||||||
redirect("/%s/_edit".format(userName))
|
redirect(s"/${userName}/_edit")
|
||||||
|
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
package app
|
package app
|
||||||
|
|
||||||
import util.Directory._
|
import util.Directory._
|
||||||
import util.UsersAuthenticator
|
import util.{JGitUtil, UsersAuthenticator}
|
||||||
import service._
|
import service._
|
||||||
import java.io.File
|
import java.io.File
|
||||||
import org.eclipse.jgit.api.Git
|
import org.eclipse.jgit.api.Git
|
||||||
@@ -56,13 +56,7 @@ trait CreateRepositoryControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
// Create the actual repository
|
// Create the actual repository
|
||||||
val gitdir = getRepositoryDir(loginUserName, form.name)
|
val gitdir = getRepositoryDir(loginUserName, form.name)
|
||||||
val repository = new RepositoryBuilder().setGitDir(gitdir).setBare.build
|
JGitUtil.initRepository(gitdir)
|
||||||
|
|
||||||
repository.create
|
|
||||||
|
|
||||||
val config = repository.getConfig
|
|
||||||
config.setBoolean("http", null, "receivepack", true)
|
|
||||||
config.save
|
|
||||||
|
|
||||||
if(form.createReadme){
|
if(form.createReadme){
|
||||||
val tmpdir = getInitRepositoryDir(loginUserName, form.name)
|
val tmpdir = getInitRepositoryDir(loginUserName, form.name)
|
||||||
@@ -73,9 +67,13 @@ trait CreateRepositoryControllerBase extends ControllerBase {
|
|||||||
// Create README.md
|
// Create README.md
|
||||||
FileUtils.writeStringToFile(new File(tmpdir, "README.md"),
|
FileUtils.writeStringToFile(new File(tmpdir, "README.md"),
|
||||||
if(form.description.nonEmpty){
|
if(form.description.nonEmpty){
|
||||||
form.name + "\n===============\n\n" + form.description.get
|
form.name + "\n" +
|
||||||
|
"===============\n" +
|
||||||
|
"\n" +
|
||||||
|
form.description.get
|
||||||
} else {
|
} else {
|
||||||
form.name + "\n===============\n"
|
form.name + "\n" +
|
||||||
|
"===============\n"
|
||||||
}, "UTF-8")
|
}, "UTF-8")
|
||||||
|
|
||||||
val git = Git.open(tmpdir)
|
val git = Git.open(tmpdir)
|
||||||
@@ -95,7 +93,7 @@ trait CreateRepositoryControllerBase extends ControllerBase {
|
|||||||
recordCreateRepositoryActivity(loginUserName, form.name, loginUserName)
|
recordCreateRepositoryActivity(loginUserName, form.name, loginUserName)
|
||||||
|
|
||||||
// redirect to the repository
|
// redirect to the repository
|
||||||
redirect("/%s/%s".format(loginUserName, form.name))
|
redirect(s"/${loginUserName}/${form.name}")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -9,10 +9,13 @@ trait IndexControllerBase extends ControllerBase { self: RepositoryService
|
|||||||
with SystemSettingsService with ActivityService =>
|
with SystemSettingsService with ActivityService =>
|
||||||
|
|
||||||
get("/"){
|
get("/"){
|
||||||
|
val loginAccount = context.loginAccount
|
||||||
|
|
||||||
html.index(getRecentActivities(),
|
html.index(getRecentActivities(),
|
||||||
getAccessibleRepositories(context.loginAccount, baseUrl),
|
getAccessibleRepositories(loginAccount, baseUrl),
|
||||||
loadSystemSettings(),
|
loadSystemSettings(),
|
||||||
context.loginAccount.map{ account => getRepositoryNamesOfUser(account.userName) }.getOrElse(Nil))
|
loginAccount.map{ account => getRepositoryNamesOfUser(account.userName) }.getOrElse(Nil)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -24,7 +24,7 @@ trait LabelsControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
post("/:owner/:repository/issues/label/new", newForm)(collaboratorsOnly { (form, repository) =>
|
post("/:owner/:repository/issues/label/new", newForm)(collaboratorsOnly { (form, repository) =>
|
||||||
createLabel(repository.owner, repository.name, form.labelName, form.color.substring(1))
|
createLabel(repository.owner, repository.name, form.labelName, form.color.substring(1))
|
||||||
redirect("/%s/%s/issues".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues")
|
||||||
})
|
})
|
||||||
|
|
||||||
ajaxGet("/:owner/:repository/issues/label/edit")(collaboratorsOnly { repository =>
|
ajaxGet("/:owner/:repository/issues/label/edit")(collaboratorsOnly { repository =>
|
||||||
@@ -53,9 +53,9 @@ trait LabelsControllerBase extends ControllerBase {
|
|||||||
private def labelName: Constraint = new Constraint(){
|
private def labelName: Constraint = new Constraint(){
|
||||||
def validate(name: String, value: String): Option[String] =
|
def validate(name: String, value: String): Option[String] =
|
||||||
if(!value.matches("^[^,]+$")){
|
if(!value.matches("^[^,]+$")){
|
||||||
Some("%s contains invalid character.".format(name))
|
Some(s"${name} contains invalid character.")
|
||||||
} else if(value.startsWith("_") || value.startsWith("-")){
|
} else if(value.startsWith("_") || value.startsWith("-")){
|
||||||
Some("%s starts with invalid character.".format(name))
|
Some(s"${name} starts with invalid character.")
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ trait MilestonesControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
post("/:owner/:repository/issues/milestones/new", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
post("/:owner/:repository/issues/milestones/new", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
||||||
createMilestone(repository.owner, repository.name, form.title, form.description, form.dueDate)
|
createMilestone(repository.owner, repository.name, form.title, form.description, form.dueDate)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/edit")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/edit")(collaboratorsOnly { repository =>
|
||||||
@@ -45,28 +45,28 @@ trait MilestonesControllerBase extends ControllerBase {
|
|||||||
post("/:owner/:repository/issues/milestones/:milestoneId/edit", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
post("/:owner/:repository/issues/milestones/:milestoneId/edit", milestoneForm)(collaboratorsOnly { (form, repository) =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
updateMilestone(milestone.copy(title = form.title, description = form.description, dueDate = form.dueDate))
|
updateMilestone(milestone.copy(title = form.title, description = form.description, dueDate = form.dueDate))
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/close")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/close")(collaboratorsOnly { repository =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
closeMilestone(milestone)
|
closeMilestone(milestone)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/open")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/open")(collaboratorsOnly { repository =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
openMilestone(milestone)
|
openMilestone(milestone)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/issues/milestones/:milestoneId/delete")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/issues/milestones/:milestoneId/delete")(collaboratorsOnly { repository =>
|
||||||
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
getMilestone(repository.owner, repository.name, params("milestoneId").toInt).map { milestone =>
|
||||||
deleteMilestone(repository.owner, repository.name, milestone.milestoneId)
|
deleteMilestone(repository.owner, repository.name, milestone.milestoneId)
|
||||||
redirect("/%s/%s/issues/milestones".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/issues/milestones")
|
||||||
} getOrElse NotFound
|
} getOrElse NotFound
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
* Redirect to the Options page.
|
* Redirect to the Options page.
|
||||||
*/
|
*/
|
||||||
get("/:owner/:repository/settings")(ownerOnly { repository =>
|
get("/:owner/:repository/settings")(ownerOnly { repository =>
|
||||||
redirect("/%s/%s/settings/options".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/options")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -47,7 +47,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
post("/:owner/:repository/settings/options", optionsForm)(ownerOnly { (form, repository) =>
|
post("/:owner/:repository/settings/options", optionsForm)(ownerOnly { (form, repository) =>
|
||||||
saveRepositoryOptions(repository.owner, repository.name, form.description, form.defaultBranch, form.isPrivate)
|
saveRepositoryOptions(repository.owner, repository.name, form.description, form.defaultBranch, form.isPrivate)
|
||||||
flash += "info" -> "Repository settings has been updated."
|
flash += "info" -> "Repository settings has been updated."
|
||||||
redirect("/%s/%s/settings/options".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/options")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -70,7 +70,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
*/
|
*/
|
||||||
post("/:owner/:repository/settings/collaborators/add", collaboratorForm)(ownerOnly { (form, repository) =>
|
post("/:owner/:repository/settings/collaborators/add", collaboratorForm)(ownerOnly { (form, repository) =>
|
||||||
addCollaborator(repository.owner, repository.name, form.userName)
|
addCollaborator(repository.owner, repository.name, form.userName)
|
||||||
redirect("/%s/%s/settings/collaborators".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/collaborators")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -78,7 +78,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
*/
|
*/
|
||||||
get("/:owner/:repository/settings/collaborators/remove")(ownerOnly { repository =>
|
get("/:owner/:repository/settings/collaborators/remove")(ownerOnly { repository =>
|
||||||
removeCollaborator(repository.owner, repository.name, params("name"))
|
removeCollaborator(repository.owner, repository.name, params("name"))
|
||||||
redirect("/%s/%s/settings/collaborators".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/settings/collaborators")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -98,7 +98,7 @@ trait RepositorySettingsControllerBase extends ControllerBase with FlashMapSuppo
|
|||||||
FileUtils.deleteDirectory(getWikiRepositoryDir(repository.owner, repository.name))
|
FileUtils.deleteDirectory(getWikiRepositoryDir(repository.owner, repository.name))
|
||||||
FileUtils.deleteDirectory(getTemporaryDir(repository.owner, repository.name))
|
FileUtils.deleteDirectory(getTemporaryDir(repository.owner, repository.name))
|
||||||
|
|
||||||
redirect("/%s".format(repository.owner))
|
redirect(s"/${repository.owner}")
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -58,13 +58,14 @@ trait RepositoryViewerControllerBase extends ControllerBase {
|
|||||||
get("/:owner/:repository/commits/:branch")(referrersOnly { repository =>
|
get("/:owner/:repository/commits/:branch")(referrersOnly { repository =>
|
||||||
val branchName = params("branch")
|
val branchName = params("branch")
|
||||||
val page = params.getOrElse("page", "1").toInt
|
val page = params.getOrElse("page", "1").toInt
|
||||||
|
|
||||||
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
val (logs, hasNext) = JGitUtil.getCommitLog(git, branchName, page, 30)
|
JGitUtil.getCommitLog(git, branchName, page, 30) match {
|
||||||
|
case Right((logs, hasNext)) =>
|
||||||
repo.html.commits(Nil, branchName, repository, logs.splitWith{ (commit1, commit2) =>
|
repo.html.commits(Nil, branchName, repository, logs.splitWith{ (commit1, commit2) =>
|
||||||
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
||||||
}, page, hasNext)
|
}, page, hasNext)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -77,12 +78,14 @@ trait RepositoryViewerControllerBase extends ControllerBase {
|
|||||||
val page = params.getOrElse("page", "1").toInt
|
val page = params.getOrElse("page", "1").toInt
|
||||||
|
|
||||||
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
val (logs, hasNext) = JGitUtil.getCommitLog(git, branchName, page, 30, path)
|
JGitUtil.getCommitLog(git, branchName, page, 30, path) match {
|
||||||
|
case Right((logs, hasNext)) =>
|
||||||
repo.html.commits(path.split("/").toList, branchName, repository,
|
repo.html.commits(path.split("/").toList, branchName, repository,
|
||||||
logs.splitWith{ (commit1, commit2) =>
|
logs.splitWith{ (commit1, commit2) =>
|
||||||
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
view.helpers.date(commit1.time) == view.helpers.date(commit2.time)
|
||||||
}, page, hasNext)
|
}, page, hasNext)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -214,27 +217,23 @@ trait RepositoryViewerControllerBase extends ControllerBase {
|
|||||||
repo.html.guide(repository)
|
repo.html.guide(repository)
|
||||||
} else {
|
} else {
|
||||||
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
|
val revisions = Seq(if(revstr.isEmpty) repository.repository.defaultBranch else revstr, repository.branchList.head)
|
||||||
// get specified commit
|
// get specified commit
|
||||||
val (revCommit, revision) = try {
|
revisions.map { rev => (git.getRepository.resolve(rev), rev)}.find(_._1 != null).map { case (objectId, revision) =>
|
||||||
val revision = if(revstr.isEmpty) repository.repository.defaultBranch else revstr
|
val revCommit = JGitUtil.getRevCommitFromId(git, objectId)
|
||||||
(JGitUtil.getRevCommitFromId(git, git.getRepository.resolve(revision)), revision)
|
|
||||||
} catch {
|
|
||||||
case e: NullPointerException => {
|
|
||||||
val revision = repository.branchList.head
|
|
||||||
(JGitUtil.getRevCommitFromId(git, git.getRepository.resolve(revision)), revision)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// get files
|
|
||||||
val files = JGitUtil.getFileList(git, revision, path)
|
|
||||||
// process README.md
|
|
||||||
val readme = files.find(_.name == "README.md").map { file =>
|
|
||||||
new String(JGitUtil.getContent(Git.open(getRepositoryDir(repository.owner, repository.name)), file.id, true).get, "UTF-8")
|
|
||||||
}
|
|
||||||
|
|
||||||
repo.html.files(revision, repository,
|
// get files
|
||||||
if(path == ".") Nil else path.split("/").toList, // current path
|
val files = JGitUtil.getFileList(git, revision, path)
|
||||||
new JGitUtil.CommitInfo(revCommit), // latest commit
|
// process README.md
|
||||||
files, readme)
|
val readme = files.find(_.name == "README.md").map { file =>
|
||||||
|
new String(JGitUtil.getContent(Git.open(getRepositoryDir(repository.owner, repository.name)), file.id, true).get, "UTF-8")
|
||||||
|
}
|
||||||
|
|
||||||
|
repo.html.files(revision, repository,
|
||||||
|
if(path == ".") Nil else path.split("/").toList, // current path
|
||||||
|
new JGitUtil.CommitInfo(revCommit), // latest commit
|
||||||
|
files, readme)
|
||||||
|
} getOrElse NotFound
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
get("/:owner/:repository/wiki")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki")(referrersOnly { repository =>
|
||||||
getWikiPage(repository.owner, repository.name, "Home").map { page =>
|
getWikiPage(repository.owner, repository.name, "Home").map { page =>
|
||||||
wiki.html.page("Home", page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
wiki.html.page("Home", page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
||||||
} getOrElse redirect("/%s/%s/wiki/Home/_edit".format(repository.owner, repository.name))
|
} getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/Home/_edit")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/:page")(referrersOnly { repository =>
|
||||||
@@ -40,14 +40,17 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
getWikiPage(repository.owner, repository.name, pageName).map { page =>
|
getWikiPage(repository.owner, repository.name, pageName).map { page =>
|
||||||
wiki.html.page(pageName, page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
wiki.html.page(pageName, page, repository, hasWritePermission(repository.owner, repository.name, context.loginAccount))
|
||||||
} getOrElse redirect("/%s/%s/wiki/%s/_edit".format(repository.owner, repository.name, pageName)) // TODO URLEncode
|
} getOrElse redirect(s"/${repository.owner}/${repository.name}/wiki/${pageName}/_edit") // TODO URLEncode
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page/_history")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/:page/_history")(referrersOnly { repository =>
|
||||||
val pageName = params("page")
|
val pageName = params("page")
|
||||||
|
|
||||||
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
wiki.html.history(Some(pageName), JGitUtil.getCommitLog(git, "master", path = pageName + ".md")._1, repository)
|
JGitUtil.getCommitLog(git, "master", path = pageName + ".md") match {
|
||||||
|
case Right((logs, hasNext)) => wiki.html.history(Some(pageName), logs, repository)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -82,7 +85,7 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
updateLastActivityDate(repository.owner, repository.name)
|
updateLastActivityDate(repository.owner, repository.name)
|
||||||
recordEditWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
recordEditWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
||||||
|
|
||||||
redirect("/%s/%s/wiki/%s".format(repository.owner, repository.name, form.pageName))
|
redirect(s"/${repository.owner}/${repository.name}/wiki/${form.pageName}")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/_new")(collaboratorsOnly {
|
get("/:owner/:repository/wiki/_new")(collaboratorsOnly {
|
||||||
@@ -98,16 +101,16 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
updateLastActivityDate(repository.owner, repository.name)
|
updateLastActivityDate(repository.owner, repository.name)
|
||||||
recordCreateWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
recordCreateWikiPageActivity(repository.owner, repository.name, loginAccount.userName, form.pageName)
|
||||||
|
|
||||||
redirect("/%s/%s/wiki/%s".format(repository.owner, repository.name, form.pageName))
|
redirect(s"/${repository.owner}/${repository.name}/wiki/${form.pageName}")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/:page/_delete")(collaboratorsOnly { repository =>
|
get("/:owner/:repository/wiki/:page/_delete")(collaboratorsOnly { repository =>
|
||||||
val pageName = params("page")
|
val pageName = params("page")
|
||||||
|
|
||||||
deleteWikiPage(repository.owner, repository.name, pageName, context.loginAccount.get.userName, "Delete %s".format(pageName))
|
deleteWikiPage(repository.owner, repository.name, pageName, context.loginAccount.get.userName, s"Delete ${pageName}")
|
||||||
updateLastActivityDate(repository.owner, repository.name)
|
updateLastActivityDate(repository.owner, repository.name)
|
||||||
|
|
||||||
redirect("/%s/%s/wiki".format(repository.owner, repository.name))
|
redirect(s"/${repository.owner}/${repository.name}/wiki")
|
||||||
})
|
})
|
||||||
|
|
||||||
get("/:owner/:repository/wiki/_pages")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/_pages")(referrersOnly { repository =>
|
||||||
@@ -117,7 +120,10 @@ trait WikiControllerBase extends ControllerBase {
|
|||||||
|
|
||||||
get("/:owner/:repository/wiki/_history")(referrersOnly { repository =>
|
get("/:owner/:repository/wiki/_history")(referrersOnly { repository =>
|
||||||
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
JGitUtil.withGit(getWikiRepositoryDir(repository.owner, repository.name)){ git =>
|
||||||
wiki.html.history(None, JGitUtil.getCommitLog(git, "master")._1, repository)
|
JGitUtil.getCommitLog(git, "master") match {
|
||||||
|
case Right((logs, hasNext)) => wiki.html.history(None, logs, repository)
|
||||||
|
case Left(_) => NotFound
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -68,12 +68,10 @@ trait WikiService {
|
|||||||
lock(owner.userName, repository){
|
lock(owner.userName, repository){
|
||||||
val dir = Directory.getWikiRepositoryDir(owner.userName, repository)
|
val dir = Directory.getWikiRepositoryDir(owner.userName, repository)
|
||||||
if(!dir.exists){
|
if(!dir.exists){
|
||||||
val repo = new RepositoryBuilder().setGitDir(dir).setBare.build
|
|
||||||
try {
|
try {
|
||||||
repo.create
|
JGitUtil.initRepository(dir)
|
||||||
saveWikiPage(owner.userName, repository, "Home", "Home", "Welcome to the %s wiki!!".format(repository), owner, "Initial Commit")
|
saveWikiPage(owner.userName, repository, "Home", "Home", s"Welcome to the ${repository} wiki!!", owner, "Initial Commit")
|
||||||
} finally {
|
} finally {
|
||||||
repo.close
|
|
||||||
// once delete cloned repository because initial cloned repository does not have 'branch.master.merge'
|
// once delete cloned repository because initial cloned repository does not have 'branch.master.merge'
|
||||||
FileUtils.deleteDirectory(Directory.getWikiWorkDir(owner.userName, repository))
|
FileUtils.deleteDirectory(Directory.getWikiWorkDir(owner.userName, repository))
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ object AutoUpdate {
|
|||||||
* If corresponding SQL file does not exist, this method do nothing.
|
* If corresponding SQL file does not exist, this method do nothing.
|
||||||
*/
|
*/
|
||||||
def update(conn: Connection): Unit = {
|
def update(conn: Connection): Unit = {
|
||||||
val sqlPath = "update/%d_%d.sql".format(majorVersion, minorVersion)
|
val sqlPath = s"update/${majorVersion}_${minorVersion}.sql"
|
||||||
val in = Thread.currentThread.getContextClassLoader.getResourceAsStream(sqlPath)
|
val in = Thread.currentThread.getContextClassLoader.getResourceAsStream(sqlPath)
|
||||||
if(in != null){
|
if(in != null){
|
||||||
val sql = IOUtils.toString(in, "UTF-8")
|
val sql = IOUtils.toString(in, "UTF-8")
|
||||||
@@ -42,7 +42,7 @@ object AutoUpdate {
|
|||||||
/**
|
/**
|
||||||
* MAJOR.MINOR
|
* MAJOR.MINOR
|
||||||
*/
|
*/
|
||||||
val versionString = "%d.%d".format(majorVersion, minorVersion)
|
val versionString = s"${majorVersion}.${minorVersion}"
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -13,13 +13,13 @@ object Directory {
|
|||||||
|
|
||||||
val GitBucketConf = new File(GitBucketHome, "gitbucket.conf")
|
val GitBucketConf = new File(GitBucketHome, "gitbucket.conf")
|
||||||
|
|
||||||
val RepositoryHome = "%s/repositories".format(GitBucketHome)
|
val RepositoryHome = s"${GitBucketHome}/repositories"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Repository names of the specified user.
|
* Repository names of the specified user.
|
||||||
*/
|
*/
|
||||||
def getRepositories(owner: String): List[String] = {
|
def getRepositories(owner: String): List[String] = {
|
||||||
val dir = new File("%s/%s".format(RepositoryHome, owner))
|
val dir = new File(s"${RepositoryHome}/${owner}")
|
||||||
if(dir.exists){
|
if(dir.exists){
|
||||||
dir.listFiles.filter { file =>
|
dir.listFiles.filter { file =>
|
||||||
file.isDirectory && !file.getName.endsWith(".wiki.git")
|
file.isDirectory && !file.getName.endsWith(".wiki.git")
|
||||||
@@ -33,24 +33,24 @@ object Directory {
|
|||||||
* Substance directory of the repository.
|
* Substance directory of the repository.
|
||||||
*/
|
*/
|
||||||
def getRepositoryDir(owner: String, repository: String): File =
|
def getRepositoryDir(owner: String, repository: String): File =
|
||||||
new File("%s/%s/%s.git".format(RepositoryHome, owner, repository))
|
new File(s"${RepositoryHome}/${owner}/${repository}.git")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Directory for uploaded files by the specified user.
|
* Directory for uploaded files by the specified user.
|
||||||
*/
|
*/
|
||||||
def getUserUploadDir(userName: String): File = new File("%s/data/%s/files".format(GitBucketHome, userName))
|
def getUserUploadDir(userName: String): File = new File(s"${GitBucketHome}/data/${userName}/files")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Root of temporary directories for the specified repository.
|
* Root of temporary directories for the specified repository.
|
||||||
*/
|
*/
|
||||||
def getTemporaryDir(owner: String, repository: String): File =
|
def getTemporaryDir(owner: String, repository: String): File =
|
||||||
new File("%s/tmp/%s/%s".format(GitBucketHome, owner, repository))
|
new File(s"${GitBucketHome}/tmp/${owner}/${repository}")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Temporary directory which is used to create an archive to download repository contents.
|
* Temporary directory which is used to create an archive to download repository contents.
|
||||||
*/
|
*/
|
||||||
def getDownloadWorkDir(owner: String, repository: String, sessionId: String): File =
|
def getDownloadWorkDir(owner: String, repository: String, sessionId: String): File =
|
||||||
new File(getTemporaryDir(owner, repository), "download/%s".format(sessionId))
|
new File(getTemporaryDir(owner, repository), s"download/${sessionId}")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Temporary directory which is used in the repository creation.
|
* Temporary directory which is used in the repository creation.
|
||||||
@@ -65,7 +65,7 @@ object Directory {
|
|||||||
* Substance directory of the wiki repository.
|
* Substance directory of the wiki repository.
|
||||||
*/
|
*/
|
||||||
def getWikiRepositoryDir(owner: String, repository: String): File =
|
def getWikiRepositoryDir(owner: String, repository: String): File =
|
||||||
new File("%s/%s/%s.wiki.git".format(Directory.RepositoryHome, owner, repository))
|
new File(s"${RepositoryHome}/${owner}/${repository}.wiki.git")
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Wiki working directory which is cloned from the wiki repository.
|
* Wiki working directory which is cloned from the wiki repository.
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ object FileUploadUtil {
|
|||||||
new SimpleDateFormat("yyyyMMddHHmmSSsss").format(new java.util.Date(System.currentTimeMillis))
|
new SimpleDateFormat("yyyyMMddHHmmSSsss").format(new java.util.Date(System.currentTimeMillis))
|
||||||
|
|
||||||
def TemporaryDir(implicit session: HttpSession): java.io.File =
|
def TemporaryDir(implicit session: HttpSession): java.io.File =
|
||||||
new java.io.File(GitBucketHome, "tmp/_upload/%s".format(session.getId))
|
new java.io.File(GitBucketHome, s"tmp/_upload/${session.getId}")
|
||||||
|
|
||||||
def getTemporaryFile(fileId: String)(implicit session: HttpSession): java.io.File =
|
def getTemporaryFile(fileId: String)(implicit session: HttpSession): java.io.File =
|
||||||
new java.io.File(TemporaryDir, fileId)
|
new java.io.File(TemporaryDir, fileId)
|
||||||
|
|||||||
@@ -153,7 +153,7 @@ object JGitUtil {
|
|||||||
}
|
}
|
||||||
|
|
||||||
RepositoryInfo(
|
RepositoryInfo(
|
||||||
owner, repository, baseUrl + "/git/%s/%s.git".format(owner, repository),
|
owner, repository, s"${baseUrl}/git/${owner}/${repository}.git",
|
||||||
// commit count
|
// commit count
|
||||||
commitCount,
|
commitCount,
|
||||||
// branches
|
// branches
|
||||||
@@ -169,7 +169,7 @@ object JGitUtil {
|
|||||||
} catch {
|
} catch {
|
||||||
// not initialized
|
// not initialized
|
||||||
case e: NoHeadException => RepositoryInfo(
|
case e: NoHeadException => RepositoryInfo(
|
||||||
owner, repository, baseUrl + "/git/%s/%s.git".format(owner, repository), 0, Nil, Nil)
|
owner, repository, s"${baseUrl}/git/${owner}/${repository}.git", 0, Nil, Nil)
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -253,7 +253,7 @@ object JGitUtil {
|
|||||||
* @param path filters by this path. default is no filter.
|
* @param path filters by this path. default is no filter.
|
||||||
* @return a tuple of the commit list and whether has next
|
* @return a tuple of the commit list and whether has next
|
||||||
*/
|
*/
|
||||||
def getCommitLog(git: Git, revision: String, page: Int = 1, limit: Int = 0, path: String = ""): (List[CommitInfo], Boolean) = {
|
def getCommitLog(git: Git, revision: String, page: Int = 1, limit: Int = 0, path: String = ""): Either[String, (List[CommitInfo], Boolean)] = {
|
||||||
val fixedPage = if(page <= 0) 1 else page
|
val fixedPage = if(page <= 0) 1 else page
|
||||||
|
|
||||||
@scala.annotation.tailrec
|
@scala.annotation.tailrec
|
||||||
@@ -267,20 +267,25 @@ object JGitUtil {
|
|||||||
}
|
}
|
||||||
|
|
||||||
val revWalk = new RevWalk(git.getRepository)
|
val revWalk = new RevWalk(git.getRepository)
|
||||||
revWalk.markStart(revWalk.parseCommit(git.getRepository.resolve(revision)))
|
val objectId = git.getRepository.resolve(revision)
|
||||||
if(path.nonEmpty){
|
if(objectId == null){
|
||||||
revWalk.setRevFilter(new RevFilter(){
|
Left(s"${revision} can't be resolved.")
|
||||||
def include(walk: RevWalk, commit: RevCommit): Boolean = {
|
} else {
|
||||||
getDiffs(git, commit.getName, false).find(_.newPath == path).nonEmpty
|
revWalk.markStart(revWalk.parseCommit(objectId))
|
||||||
}
|
if(path.nonEmpty){
|
||||||
override def clone(): RevFilter = this
|
revWalk.setRevFilter(new RevFilter(){
|
||||||
})
|
def include(walk: RevWalk, commit: RevCommit): Boolean = {
|
||||||
|
getDiffs(git, commit.getName, false).find(_.newPath == path).nonEmpty
|
||||||
|
}
|
||||||
|
override def clone(): RevFilter = this
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
val commits = getCommitLog(revWalk.iterator, 0, Nil)
|
||||||
|
revWalk.release
|
||||||
|
|
||||||
|
Right(commits)
|
||||||
}
|
}
|
||||||
|
|
||||||
val commits = getCommitLog(revWalk.iterator, 0, Nil)
|
|
||||||
revWalk.release
|
|
||||||
|
|
||||||
commits
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -502,4 +507,20 @@ object JGitUtil {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def initRepository(dir: java.io.File): Unit = {
|
||||||
|
val repository = new RepositoryBuilder().setGitDir(dir).setBare.build
|
||||||
|
try {
|
||||||
|
repository.create
|
||||||
|
setReceivePack(repository)
|
||||||
|
} finally {
|
||||||
|
repository.close
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private def setReceivePack(repository: org.eclipse.jgit.lib.Repository): Unit = {
|
||||||
|
val config = repository.getConfig
|
||||||
|
config.setBoolean("http", null, "receivepack", true)
|
||||||
|
config.save
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -11,9 +11,9 @@ trait Validations {
|
|||||||
def identifier: Constraint = new Constraint(){
|
def identifier: Constraint = new Constraint(){
|
||||||
def validate(name: String, value: String): Option[String] =
|
def validate(name: String, value: String): Option[String] =
|
||||||
if(!value.matches("^[a-zA-Z0-9\\-_]+$")){
|
if(!value.matches("^[a-zA-Z0-9\\-_]+$")){
|
||||||
Some("%s contains invalid character.".format(name))
|
Some(s"${name} contains invalid character.")
|
||||||
} else if(value.startsWith("_") || value.startsWith("-")){
|
} else if(value.startsWith("_") || value.startsWith("-")){
|
||||||
Some("%s starts with invalid character.".format(name))
|
Some(s"${name} starts with invalid character.")
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -101,7 +101,7 @@ class GitBucketHtmlSerializer(
|
|||||||
override def visit(node: TextNode) {
|
override def visit(node: TextNode) {
|
||||||
// convert commit id to link.
|
// convert commit id to link.
|
||||||
val text = if(enableCommitLink) node.getText.replaceAll("(^|\\W)([0-9a-f]{40})(\\W|$)",
|
val text = if(enableCommitLink) node.getText.replaceAll("(^|\\W)([0-9a-f]{40})(\\W|$)",
|
||||||
"<a href=\"%s/%s/%s/commit/$2\">$2</a>".format(context.path, repository.owner, repository.name))
|
s"""<a href="${context.path}/${repository.owner}/${repository.name}/commit/$$2">$$2</a>""")
|
||||||
else node.getText
|
else node.getText
|
||||||
|
|
||||||
if (abbreviations.isEmpty) {
|
if (abbreviations.isEmpty) {
|
||||||
@@ -116,7 +116,7 @@ class GitBucketHtmlSerializer(
|
|||||||
if(enableIssueLink && text.matches("#[\\d]+")){
|
if(enableIssueLink && text.matches("#[\\d]+")){
|
||||||
// convert issue id to link
|
// convert issue id to link
|
||||||
val issueId = text.substring(1).toInt
|
val issueId = text.substring(1).toInt
|
||||||
printer.print("<a href=\"%s/%s/%s/issues/%d\">#%d</a>".format(context.path, repository.owner, repository.name, issueId, issueId))
|
printer.print(s"""<a href="${context.path}/${repository.owner}/${repository.name}/issues/${issueId}">#${issueId}</a>""")
|
||||||
} else {
|
} else {
|
||||||
printTag(node, "h" + node.getLevel)
|
printTag(node, "h" + node.getLevel)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,9 +35,7 @@ object helpers {
|
|||||||
Html(Markdown.toHtml(value, repository, enableWikiLink, enableCommitLink, enableIssueLink))
|
Html(Markdown.toHtml(value, repository, enableWikiLink, enableCommitLink, enableIssueLink))
|
||||||
}
|
}
|
||||||
|
|
||||||
def activityMessage(message: String)(implicit context: app.Context): Html = {
|
def activityMessage(message: String)(implicit context: app.Context): Html =
|
||||||
val a = s"a $message aa $$1 a"
|
|
||||||
|
|
||||||
Html(message
|
Html(message
|
||||||
.replaceAll("\\[issue:([^\\s]+?)/([^\\s]+?)#((\\d+))\\]" , s"""<a href="${context.path}/$$1/$$2/issues/$$3">$$1/$$2#$$3</a>""")
|
.replaceAll("\\[issue:([^\\s]+?)/([^\\s]+?)#((\\d+))\\]" , s"""<a href="${context.path}/$$1/$$2/issues/$$3">$$1/$$2#$$3</a>""")
|
||||||
.replaceAll("\\[repo:([^\\s]+?)/([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2\">$$1/$$2</a>""")
|
.replaceAll("\\[repo:([^\\s]+?)/([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2\">$$1/$$2</a>""")
|
||||||
@@ -45,23 +43,24 @@ object helpers {
|
|||||||
.replaceAll("\\[tag:([^\\s]+?)/([^\\s]+?)#([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2/tree/$$3">$$3</a>""")
|
.replaceAll("\\[tag:([^\\s]+?)/([^\\s]+?)#([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1/$$2/tree/$$3">$$3</a>""")
|
||||||
.replaceAll("\\[user:([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1">$$1</a>""")
|
.replaceAll("\\[user:([^\\s]+?)\\]" , s"""<a href="${context.path}/$$1">$$1</a>""")
|
||||||
)
|
)
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generates the url to the repository.
|
* Generates the url to the repository.
|
||||||
*/
|
*/
|
||||||
def url(repository: service.RepositoryService.RepositoryInfo)(implicit context: app.Context): String =
|
def url(repository: service.RepositoryService.RepositoryInfo)(implicit context: app.Context): String =
|
||||||
"%s/%s/%s".format(context.path, repository.owner, repository.name)
|
s"${context.path}/${repository.owner}/${repository.name}"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generates the url to the account page.
|
* Generates the url to the account page.
|
||||||
*/
|
*/
|
||||||
def url(userName: String)(implicit context: app.Context): String = "%s/%s".format(context.path, userName)
|
def url(userName: String)(implicit context: app.Context): String =
|
||||||
|
s"${context.path}/${userName}"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the url to the root of assets.
|
* Returns the url to the root of assets.
|
||||||
*/
|
*/
|
||||||
def assets(implicit context: app.Context): String = "%s/assets".format(context.path)
|
def assets(implicit context: app.Context): String =
|
||||||
|
s"${context.path}/assets"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts issue id and commit id to link.
|
* Converts issue id and commit id to link.
|
||||||
@@ -71,9 +70,9 @@ object helpers {
|
|||||||
// escape HTML tags
|
// escape HTML tags
|
||||||
.replaceAll("&", "&").replaceAll("<", "<").replaceAll(">", ">").replaceAll("\"", """)
|
.replaceAll("&", "&").replaceAll("<", "<").replaceAll(">", ">").replaceAll("\"", """)
|
||||||
// convert issue id to link
|
// convert issue id to link
|
||||||
.replaceAll("(^|\\W)#(\\d+)(\\W|$)", "$1<a href=\"%s/%s/%s/issues/$2\">#$2</a>$3".format(context.path, repository.owner, repository.name))
|
.replaceAll("(^|\\W)#(\\d+)(\\W|$)", s"""$$1<a href="${context.path}/${repository.owner}/${repository.name}/issues/$$2">#$$2</a>$$3""")
|
||||||
// convert commit id to link
|
// convert commit id to link
|
||||||
.replaceAll("(^|\\W)([a-f0-9]{40})(\\W|$)", "$1<a href=\"%s/%s/%s/commit/$2\">$2</a>$3").format(context.path, repository.owner, repository.name))
|
.replaceAll("(^|\\W)([a-f0-9]{40})(\\W|$)", s"""$$1<a href="${context.path}/${repository.owner}/${repository.name}/commit/$$2">$$2</a>$$3"""))
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -14,10 +14,10 @@
|
|||||||
@activity.additionalInfo.map { additionalInfo =>
|
@activity.additionalInfo.map { additionalInfo =>
|
||||||
@(activity.activityType match {
|
@(activity.activityType match {
|
||||||
case "create_wiki" => {
|
case "create_wiki" => {
|
||||||
<div class="small activity-message">Created <a href={"%s/%s/%s/wiki/%s".format(path, activity.userName, activity.repositoryName, additionalInfo)}>{additionalInfo}</a>.</div>
|
<div class="small activity-message">Created <a href={s"${path}/${activity.userName}/${activity.repositoryName}/wiki/${additionalInfo}"}>{additionalInfo}</a>.</div>
|
||||||
}
|
}
|
||||||
case "edit_wiki" => {
|
case "edit_wiki" => {
|
||||||
<div class="small activity-message">Edited <a href={"%s/%s/%s/wiki/%s".format(path, activity.userName, activity.repositoryName, additionalInfo)}>{additionalInfo}</a>.</div>
|
<div class="small activity-message">Edited <a href={s"${path}/${activity.userName}/${activity.repositoryName}/wiki/${additionalInfo}"}>{additionalInfo}</a>.</div>
|
||||||
}
|
}
|
||||||
case "push" => {
|
case "push" => {
|
||||||
<div class="small activity-message">
|
<div class="small activity-message">
|
||||||
@@ -26,7 +26,7 @@
|
|||||||
<div>...</div>
|
<div>...</div>
|
||||||
} else {
|
} else {
|
||||||
<div>
|
<div>
|
||||||
<a href={"%s/%s/%s/commit/%s".format(path, activity.userName, activity.repositoryName, commit.substring(0, 40))} class="monospace">{commit.substring(0, 7)}</a>
|
<a href={s"${path}/${activity.userName}/${activity.repositoryName}/commit/${commit.substring(0, 40)}"} class="monospace">{commit.substring(0, 7)}</a>
|
||||||
<span>{commit.substring(41)}</span>
|
<span>{commit.substring(41)}</span>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user