diff options
author | Unknwon <u@gogs.io> | 2016-08-30 05:07:50 -0700 |
---|---|---|
committer | Unknwon <u@gogs.io> | 2016-08-30 05:23:59 -0700 |
commit | 643142acab44d46aa6c001c90ad5d307a8662b99 (patch) | |
tree | 8e26f8959af5175e865d71a35df5dc087ff14dc8 /routers | |
parent | 7c31f235da287b0ba1499986758332c2c346deb0 (diff) | |
download | gitea-643142acab44d46aa6c001c90ad5d307a8662b99.tar.gz gitea-643142acab44d46aa6c001c90ad5d307a8662b99.zip |
Web editor: support upload files
Diffstat (limited to 'routers')
-rw-r--r-- | routers/repo/editor.go | 199 | ||||
-rw-r--r-- | routers/repo/issue.go | 2 | ||||
-rw-r--r-- | routers/repo/upload.go | 253 | ||||
-rw-r--r-- | routers/repo/view.go | 7 |
4 files changed, 186 insertions, 275 deletions
diff --git a/routers/repo/editor.go b/routers/repo/editor.go index 465de00214..f471e5ee22 100644 --- a/routers/repo/editor.go +++ b/routers/repo/editor.go @@ -5,7 +5,9 @@ package repo import ( + "fmt" "io/ioutil" + "net/http" "path" "strings" @@ -23,6 +25,7 @@ const ( EDIT_FILE base.TplName = "repo/editor/edit" EDIT_DIFF_PREVIEW base.TplName = "repo/editor/diff_preview" DELETE_FILE base.TplName = "repo/editor/delete" + UPLOAD_FILE base.TplName = "repo/editor/upload" ) func editFile(ctx *context.Context, isNewFile bool) { @@ -31,8 +34,6 @@ func editFile(ctx *context.Context, isNewFile bool) { ctx.Data["RequireHighlightJS"] = true ctx.Data["RequireSimpleMDE"] = true - branchLink := ctx.Repo.RepoLink + "/src/" + ctx.Repo.BranchName - var treeNames []string if len(ctx.Repo.TreePath) > 0 { treeNames = strings.Split(ctx.Repo.TreePath, "/") @@ -41,11 +42,7 @@ func editFile(ctx *context.Context, isNewFile bool) { if !isNewFile { entry, err := ctx.Repo.Commit.GetTreeEntryByPath(ctx.Repo.TreePath) if err != nil { - if git.IsErrNotExist(err) { - ctx.Handle(404, "GetTreeEntryByPath", err) - } else { - ctx.Handle(500, "GetTreeEntryByPath", err) - } + ctx.NotFoundOrServerError("GetTreeEntryByPath", git.IsErrNotExist, err) return } @@ -91,9 +88,8 @@ func editFile(ctx *context.Context, isNewFile bool) { treeNames = append(treeNames, "") // Append empty string to allow user name the new file. } - ctx.Data["TreePath"] = ctx.Repo.TreePath ctx.Data["TreeNames"] = treeNames - ctx.Data["BranchLink"] = branchLink + ctx.Data["BranchLink"] = ctx.Repo.RepoLink + "/src/" + ctx.Repo.BranchName ctx.Data["commit_summary"] = "" ctx.Data["commit_message"] = "" ctx.Data["commit_choice"] = "direct" @@ -122,7 +118,6 @@ func editFilePost(ctx *context.Context, form auth.EditRepoFileForm, isNewFile bo oldBranchName := ctx.Repo.BranchName branchName := oldBranchName - branchLink := ctx.Repo.RepoLink + "/src/" + branchName oldTreePath := ctx.Repo.TreePath lastCommit := form.LastCommit form.LastCommit = ctx.Repo.Commit.ID.String() @@ -140,7 +135,7 @@ func editFilePost(ctx *context.Context, form auth.EditRepoFileForm, isNewFile bo ctx.Data["TreePath"] = form.TreePath ctx.Data["TreeNames"] = treeNames - ctx.Data["BranchLink"] = branchLink + ctx.Data["BranchLink"] = ctx.Repo.RepoLink + "/src/" + branchName ctx.Data["FileContent"] = form.Content ctx.Data["commit_summary"] = form.CommitSummary ctx.Data["commit_message"] = form.CommitMessage @@ -180,7 +175,7 @@ func editFilePost(ctx *context.Context, form auth.EditRepoFileForm, isNewFile bo break } - ctx.Handle(500, "GetTreeEntryByPath", err) + ctx.Handle(500, "Repo.Commit.GetTreeEntryByPath", err) return } if index != len(treeNames)-1 { @@ -326,7 +321,6 @@ func DeleteFilePost(ctx *context.Context, form auth.DeleteRepoFileForm) { oldBranchName := ctx.Repo.BranchName branchName := oldBranchName - treePath := ctx.Repo.TreePath if form.CommitChoice == "commit-to-new-branch" { branchName = form.NewBranchName @@ -351,7 +345,7 @@ func DeleteFilePost(ctx *context.Context, form auth.DeleteRepoFileForm) { message := strings.TrimSpace(form.CommitSummary) if len(message) == 0 { - message = ctx.Tr("repo.editor.delete", treePath) + message = ctx.Tr("repo.editor.delete", ctx.Repo.TreePath) } form.CommitMessage = strings.TrimSpace(form.CommitMessage) @@ -363,13 +357,186 @@ func DeleteFilePost(ctx *context.Context, form auth.DeleteRepoFileForm) { LastCommitID: ctx.Repo.CommitID, OldBranch: oldBranchName, NewBranch: branchName, - TreePath: treePath, + TreePath: ctx.Repo.TreePath, Message: message, }); err != nil { ctx.Handle(500, "DeleteRepoFile", err) return } - ctx.Flash.Success(ctx.Tr("repo.editor.file_delete_success", treePath)) + ctx.Flash.Success(ctx.Tr("repo.editor.file_delete_success", ctx.Repo.TreePath)) ctx.Redirect(ctx.Repo.RepoLink + "/src/" + branchName) } + +func renderUploadSettings(ctx *context.Context) { + ctx.Data["RequireDropzone"] = true + ctx.Data["UploadAllowedTypes"] = strings.Join(setting.Repository.Upload.AllowedTypes, ",") + ctx.Data["UploadMaxSize"] = setting.Repository.Upload.FileMaxSize + ctx.Data["UploadMaxFiles"] = setting.Repository.Upload.MaxFiles +} + +func UploadFile(ctx *context.Context) { + ctx.Data["PageIsUpload"] = true + renderUploadSettings(ctx) + + // We must at least have one element for user to input. + treeNames := []string{""} + if len(ctx.Repo.TreePath) > 0 { + treeNames = strings.Split(ctx.Repo.TreePath, "/") + } + + ctx.Data["TreeNames"] = treeNames + ctx.Data["BranchLink"] = ctx.Repo.RepoLink + "/src/" + ctx.Repo.BranchName + ctx.Data["commit_summary"] = "" + ctx.Data["commit_message"] = "" + ctx.Data["commit_choice"] = "direct" + ctx.Data["new_branch_name"] = "" + + ctx.HTML(200, UPLOAD_FILE) +} + +func UploadFilePost(ctx *context.Context, form auth.UploadRepoFileForm) { + ctx.Data["PageIsUpload"] = true + renderUploadSettings(ctx) + + oldBranchName := ctx.Repo.BranchName + branchName := oldBranchName + + if form.CommitChoice == "commit-to-new-branch" { + branchName = form.NewBranchName + } + + form.TreePath = strings.Trim(form.TreePath, " /") + + // We must at least have one element for user to input. + treeNames := []string{""} + if len(form.TreePath) > 0 { + treeNames = strings.Split(form.TreePath, "/") + } + + ctx.Data["TreePath"] = form.TreePath + ctx.Data["TreeNames"] = treeNames + ctx.Data["BranchLink"] = ctx.Repo.RepoLink + "/src/" + branchName + ctx.Data["commit_summary"] = form.CommitSummary + ctx.Data["commit_message"] = form.CommitMessage + ctx.Data["commit_choice"] = form.CommitChoice + ctx.Data["new_branch_name"] = branchName + + if ctx.HasError() { + ctx.HTML(200, UPLOAD_FILE) + return + } + + if oldBranchName != branchName { + if _, err := ctx.Repo.Repository.GetBranch(branchName); err == nil { + ctx.Data["Err_NewBranchName"] = true + ctx.RenderWithErr(ctx.Tr("repo.editor.branch_already_exists", branchName), UPLOAD_FILE, &form) + return + } + } + + var newTreePath string + for _, part := range treeNames { + newTreePath = path.Join(newTreePath, part) + entry, err := ctx.Repo.Commit.GetTreeEntryByPath(newTreePath) + if err != nil { + if git.IsErrNotExist(err) { + // Means there is no item with that name, so we're good + break + } + + ctx.Handle(500, "Repo.Commit.GetTreeEntryByPath", err) + return + } + + // User can only upload files to a directory. + if !entry.IsDir() { + ctx.Data["Err_TreePath"] = true + ctx.RenderWithErr(ctx.Tr("repo.editor.directory_is_a_file", part), UPLOAD_FILE, &form) + return + } + } + + message := strings.TrimSpace(form.CommitSummary) + if len(message) == 0 { + message = ctx.Tr("repo.editor.upload_files_to_dir", form.TreePath) + } + + form.CommitMessage = strings.TrimSpace(form.CommitMessage) + if len(form.CommitMessage) > 0 { + message += "\n\n" + form.CommitMessage + } + + if err := ctx.Repo.Repository.UploadRepoFiles(ctx.User, models.UploadRepoFileOptions{ + LastCommitID: ctx.Repo.CommitID, + OldBranch: oldBranchName, + NewBranch: branchName, + TreePath: form.TreePath, + Message: message, + Files: form.Files, + }); err != nil { + ctx.Data["Err_TreePath"] = true + ctx.RenderWithErr(ctx.Tr("repo.editor.unable_to_upload_files", form.TreePath, err), UPLOAD_FILE, &form) + return + } + + ctx.Redirect(ctx.Repo.RepoLink + "/src/" + branchName + "/" + form.TreePath) +} + +func UploadFileToServer(ctx *context.Context) { + file, header, err := ctx.Req.FormFile("file") + if err != nil { + ctx.Error(500, fmt.Sprintf("FormFile: %v", err)) + return + } + defer file.Close() + + buf := make([]byte, 1024) + n, _ := file.Read(buf) + if n > 0 { + buf = buf[:n] + } + fileType := http.DetectContentType(buf) + + if len(setting.Repository.Upload.AllowedTypes) > 0 { + allowed := false + for _, t := range setting.Repository.Upload.AllowedTypes { + t := strings.Trim(t, " ") + if t == "*/*" || t == fileType { + allowed = true + break + } + } + + if !allowed { + ctx.Error(400, ErrFileTypeForbidden.Error()) + return + } + } + + upload, err := models.NewUpload(header.Filename, buf, file) + if err != nil { + ctx.Error(500, fmt.Sprintf("NewUpload: %v", err)) + return + } + + log.Trace("New file uploaded: %s", upload.UUID) + ctx.JSON(200, map[string]string{ + "uuid": upload.UUID, + }) +} + +func RemoveUploadFileFromServer(ctx *context.Context, form auth.RemoveUploadFileForm) { + if len(form.File) == 0 { + ctx.Status(204) + return + } + + if err := models.DeleteUploadByUUID(form.File); err != nil { + ctx.Error(500, fmt.Sprintf("DeleteUploadByUUID: %v", err)) + return + } + + log.Trace("Upload file removed: %s", form.File) + ctx.Status(204) +} diff --git a/routers/repo/issue.go b/routers/repo/issue.go index a5908c182f..e2d8000e52 100644 --- a/routers/repo/issue.go +++ b/routers/repo/issue.go @@ -447,7 +447,6 @@ func UploadIssueAttachment(ctx *context.Context) { return } - allowedTypes := strings.Split(setting.AttachmentAllowedTypes, ",") file, header, err := ctx.Req.FormFile("file") if err != nil { ctx.Error(500, fmt.Sprintf("FormFile: %v", err)) @@ -462,6 +461,7 @@ func UploadIssueAttachment(ctx *context.Context) { } fileType := http.DetectContentType(buf) + allowedTypes := strings.Split(setting.AttachmentAllowedTypes, ",") allowed := false for _, t := range allowedTypes { t := strings.Trim(t, " ") diff --git a/routers/repo/upload.go b/routers/repo/upload.go deleted file mode 100644 index 5e0f91668f..0000000000 --- a/routers/repo/upload.go +++ /dev/null @@ -1,253 +0,0 @@ -// Copyright 2016 The Gogs Authors. All rights reserved. -// Use of this source code is governed by a MIT-style -// license that can be found in the LICENSE file. - -package repo - -import ( - "fmt" - "net/http" - "path" - "strings" - - git "github.com/gogits/git-module" - - "github.com/gogits/gogs/models" - "github.com/gogits/gogs/modules/auth" - "github.com/gogits/gogs/modules/base" - "github.com/gogits/gogs/modules/context" - "github.com/gogits/gogs/modules/log" - "github.com/gogits/gogs/modules/setting" -) - -const ( - UPLOAD base.TplName = "repo/upload" -) - -func renderUploadSettings(ctx *context.Context) { - ctx.Data["RequireDropzone"] = true - ctx.Data["IsUploadEnabled"] = setting.Repository.Upload.Enabled - ctx.Data["UploadAllowedTypes"] = strings.Join(setting.Repository.Upload.AllowedTypes, ",") - ctx.Data["UploadMaxSize"] = setting.Repository.Upload.FileMaxSize - ctx.Data["UploadMaxFiles"] = setting.Repository.Upload.MaxFiles -} - -func UploadFile(ctx *context.Context) { - ctx.Data["PageIsUpload"] = true - - userName := ctx.Repo.Owner.Name - repoName := ctx.Repo.Repository.Name - branchName := ctx.Repo.BranchName - branchLink := ctx.Repo.RepoLink + "/src/" + branchName - treeName := ctx.Repo.TreePath - - treeNames := []string{""} - if len(treeName) > 0 { - treeNames = strings.Split(treeName, "/") - } - - ctx.Data["UserName"] = userName - ctx.Data["RepoName"] = repoName - ctx.Data["BranchName"] = branchName - ctx.Data["TreeName"] = treeName - ctx.Data["TreeNames"] = treeNames - ctx.Data["BranchLink"] = branchLink - ctx.Data["CommitSummary"] = "" - ctx.Data["CommitMessage"] = "" - ctx.Data["CommitChoice"] = "direct" - ctx.Data["NewBranchName"] = "" - ctx.Data["CommitDirectlyToThisBranch"] = ctx.Tr("repo.commit_directly_to_this_branch", "<strong class=\"branch-name\">"+branchName+"</strong>") - ctx.Data["CreateNewBranch"] = ctx.Tr("repo.create_new_branch", "<strong>"+ctx.Tr("repo.new_branch")+"</strong>") - renderUploadSettings(ctx) - - ctx.HTML(200, UPLOAD) -} - -func UploadFilePost(ctx *context.Context, form auth.UploadRepoFileForm) { - ctx.Data["PageIsUpload"] = true - renderUploadSettings(ctx) - - userName := ctx.Repo.Owner.Name - repoName := ctx.Repo.Repository.Name - oldBranchName := ctx.Repo.BranchName - branchName := oldBranchName - branchLink := ctx.Repo.RepoLink + "/src/" + branchName - commitChoice := form.CommitChoice - files := form.Files - - if commitChoice == "commit-to-new-branch" { - branchName = form.NewBranchName - } - - treeName := form.TreeName - treeName = strings.Trim(treeName, " ") - treeName = strings.Trim(treeName, "/") - - treeNames := []string{""} - if len(treeName) > 0 { - treeNames = strings.Split(treeName, "/") - } - - ctx.Data["UserName"] = userName - ctx.Data["RepoName"] = repoName - ctx.Data["BranchName"] = branchName - ctx.Data["TreeName"] = treeName - ctx.Data["TreeNames"] = treeNames - ctx.Data["BranchLink"] = branchLink - ctx.Data["CommitSummary"] = form.CommitSummary - ctx.Data["CommitMessage"] = form.CommitMessage - ctx.Data["CommitChoice"] = commitChoice - ctx.Data["NewBranchName"] = branchName - ctx.Data["CommitDirectlyToThisBranch"] = ctx.Tr("repo.commit_directly_to_this_branch", "<strong class=\"branch-name\">"+oldBranchName+"</strong>") - ctx.Data["CreateNewBranch"] = ctx.Tr("repo.create_new_branch", "<strong>"+ctx.Tr("repo.new_branch")+"</strong>") - - if ctx.HasError() { - ctx.HTML(200, UPLOAD) - return - } - - if oldBranchName != branchName { - if _, err := ctx.Repo.Repository.GetBranch(branchName); err == nil { - ctx.Data["Err_Branchname"] = true - ctx.RenderWithErr(ctx.Tr("repo.branch_already_exists"), UPLOAD, &form) - log.Error(4, "%s: %s - %s", "BranchName", branchName, "Branch already exists") - return - } - - } - - treepath := "" - for _, part := range treeNames { - treepath = path.Join(treepath, part) - entry, err := ctx.Repo.Commit.GetTreeEntryByPath(treepath) - if err != nil { - // Means there is no item with that name, so we're good - break - } - if !entry.IsDir() { - ctx.Data["Err_Filename"] = true - ctx.RenderWithErr(ctx.Tr("repo.directory_is_a_file"), UPLOAD, &form) - log.Error(4, "%s: %s - %s", "UploadFile", treeName, "Directory given is a file") - return - } - } - - message := "" - if form.CommitSummary != "" { - message = strings.Trim(form.CommitSummary, " ") - } else { - message = ctx.Tr("repo.add_files_to_dir", "'"+treeName+"'") - } - if strings.Trim(form.CommitMessage, " ") != "" { - message += "\n\n" + strings.Trim(form.CommitMessage, " ") - } - - if err := ctx.Repo.Repository.UploadRepoFiles(ctx.User, oldBranchName, branchName, treeName, message, files); err != nil { - ctx.Data["Err_Directory"] = true - ctx.RenderWithErr(ctx.Tr("repo.unable_to_upload_files"), UPLOAD, &form) - log.Error(4, "%s: %v", "UploadFile", err) - return - } - - // Was successful, so now need to call models.CommitRepoAction() with the new commitID for webhooks and watchers - if branch, err := ctx.Repo.Repository.GetBranch(branchName); err != nil { - log.Error(4, "repo.Repository.GetBranch(%s): %v", branchName, err) - } else if commit, err := branch.GetCommit(); err != nil { - log.Error(4, "branch.GetCommit(): %v", err) - } else { - pc := &models.PushCommits{ - Len: 1, - Commits: []*models.PushCommit{models.CommitToPushCommit(commit)}, - } - oldCommitID := ctx.Repo.CommitID - newCommitID := commit.ID.String() - if branchName != oldBranchName { - oldCommitID = "0000000000000000000000000000000000000000" // New Branch so we use all 0s - } - if err := models.CommitRepoAction(models.CommitRepoActionOptions{ - PusherName: ctx.User.Name, - RepoOwnerID: ctx.Repo.Owner.ID, - RepoName: ctx.Repo.Owner.Name, - RefFullName: git.BRANCH_PREFIX + branchName, - OldCommitID: oldCommitID, - NewCommitID: newCommitID, - Commits: pc, - }); err != nil { - log.Error(4, "models.CommitRepoAction(branch = %s): %v", branchName, err) - } - } - - ctx.Redirect(ctx.Repo.RepoLink + "/src/" + branchName + "/" + treeName) -} - -func UploadFileToServer(ctx *context.Context) { - if !setting.Repository.Upload.Enabled { - ctx.Error(404, "upload is not enabled") - return - } - - file, header, err := ctx.Req.FormFile("file") - if err != nil { - ctx.Error(500, fmt.Sprintf("FormFile: %v", err)) - return - } - defer file.Close() - - buf := make([]byte, 1024) - n, _ := file.Read(buf) - if n > 0 { - buf = buf[:n] - } - fileType := http.DetectContentType(buf) - - if len(setting.Repository.Upload.AllowedTypes) > 0 { - allowed := false - for _, t := range setting.Repository.Upload.AllowedTypes { - t := strings.Trim(t, " ") - if t == "*/*" || t == fileType { - allowed = true - break - } - } - - if !allowed { - ctx.Error(400, ErrFileTypeForbidden.Error()) - return - } - } - - up, err := models.NewUpload(header.Filename, buf, file, ctx.User.ID, ctx.Repo.Repository.ID) - if err != nil { - ctx.Error(500, fmt.Sprintf("NewUpload: %v", err)) - return - } - - log.Trace("New file uploaded: %s", up.UUID) - ctx.JSON(200, map[string]string{ - "uuid": up.UUID, - }) -} - -func RemoveUploadFileFromServer(ctx *context.Context, form auth.RemoveUploadFileForm) { - if !setting.Repository.Upload.Enabled { - ctx.Error(404, "upload is not enabled") - return - } - - if len(form.File) == 0 { - ctx.Error(404, "invalid params") - return - } - - uuid := form.File - - if err := models.RemoveUpload(uuid, ctx.User.ID, ctx.Repo.Repository.ID); err != nil { - ctx.Error(500, fmt.Sprintf("RemoveUpload: %v", err)) - return - } - - log.Trace("Upload file removed: %s", uuid) - ctx.JSON(200, map[string]string{ - "uuid": uuid, - }) -} diff --git a/routers/repo/view.go b/routers/repo/view.go index 42bf62c606..055785564d 100644 --- a/routers/repo/view.go +++ b/routers/repo/view.go @@ -113,10 +113,7 @@ func renderDirectory(ctx *context.Context, treeLink string) { // Check permission to add or upload new file. if ctx.Repo.IsWriter() && ctx.Repo.IsViewBranch { ctx.Data["CanAddFile"] = true - // uploadFileLink := ctx.Repo.RepoLink + "/upload/" + branchName - // if setting.Repository.Upload.Enabled { - // ctx.Data["UploadFileLink"] = uploadFileLink + "/" + ctx.Repo.TreePath - // } + ctx.Data["CanUploadFile"] = setting.Repository.Upload.Enabled } } @@ -253,7 +250,7 @@ func Home(ctx *context.Context) { ec, err := ctx.Repo.GetEditorconfig() if err != nil && !git.IsErrNotExist(err) { - ctx.Handle(500, "ErrGettingEditorconfig", err) + ctx.Handle(500, "Repo.GetEditorconfig", err) return } ctx.Data["Editorconfig"] = ec |