package app import util.Directory._ import util.{JGitUtil, UsersAuthenticator} import service._ import java.io.File import org.eclipse.jgit.api.Git import org.apache.commons.io._ import jp.sf.amateras.scalatra.forms._ class CreateRepositoryController extends CreateRepositoryControllerBase with RepositoryService with AccountService with WikiService with LabelsService with ActivityService with UsersAuthenticator /** * Creates new repository. */ trait CreateRepositoryControllerBase extends ControllerBase { self: RepositoryService with AccountService with WikiService with LabelsService with ActivityService with UsersAuthenticator => case class RepositoryCreationForm(owner: String, name: String, description: Option[String], isPrivate: Boolean, createReadme: Boolean) val form = mapping( "owner" -> trim(label("Owner" , text(required, maxlength(40), identifier, existsAccount))), "name" -> trim(label("Repository name", text(required, maxlength(40), identifier, unique))), "description" -> trim(label("Description" , optional(text()))), "isPrivate" -> trim(label("Repository Type", boolean())), "createReadme" -> trim(label("Create README" , boolean())) )(RepositoryCreationForm.apply) /** * Show the new repository form. */ get("/new")(usersOnly { html.newrepo(getGroupsByUserName(context.loginAccount.get.userName)) }) /** * Create new repository. */ post("/new", form)(usersOnly { form => val ownerAccount = getAccountByUserName(form.owner).get val loginAccount = context.loginAccount.get val loginUserName = loginAccount.userName // Insert to the database at first createRepository(form.name, form.owner, form.description, form.isPrivate) // Add collaborators for group repository if(ownerAccount.isGroupAccount){ getGroupMembers(form.owner).foreach { userName => addCollaborator(form.owner, form.name, userName) } } // Insert default labels createLabel(form.owner, form.name, "bug", "fc2929") createLabel(form.owner, form.name, "duplicate", "cccccc") createLabel(form.owner, form.name, "enhancement", "84b6eb") createLabel(form.owner, form.name, "invalid", "e6e6e6") createLabel(form.owner, form.name, "question", "cc317c") createLabel(form.owner, form.name, "wontfix", "ffffff") // Create the actual repository val gitdir = getRepositoryDir(form.owner, form.name) JGitUtil.initRepository(gitdir) if(form.createReadme){ val tmpdir = getInitRepositoryDir(form.owner, form.name) try { // Clone the repository Git.cloneRepository.setURI(gitdir.toURI.toString).setDirectory(tmpdir).call // Create README.md FileUtils.writeStringToFile(new File(tmpdir, "README.md"), if(form.description.nonEmpty){ form.name + "\n" + "===============\n" + "\n" + form.description.get } else { form.name + "\n" + "===============\n" }, "UTF-8") val git = Git.open(tmpdir) git.add.addFilepattern("README.md").call git.commit.setCommitter(loginAccount.userName, loginAccount.mailAddress).setMessage("Initial commit").call git.push.call } finally { FileUtils.deleteDirectory(tmpdir) } } // Create Wiki repository createWikiRepository(loginAccount, form.owner, form.name) // Record activity recordCreateRepositoryActivity(form.owner, form.name, loginUserName) // redirect to the repository redirect(s"/${form.owner}/${form.name}") }) private def existsAccount: Constraint = new Constraint(){ def validate(name: String, value: String): Option[String] = if(getAccountByUserName(value).isEmpty) Some("User or group does not exist.") else None } /** * Duplicate check for the repository name. */ private def unique: Constraint = new Constraint(){ def validate(name: String, value: String): Option[String] = params.get("owner").flatMap { userName => getRepositoryNamesOfUser(userName).find(_ == value).map(_ => "Repository already exists.") } } }