Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
584 lines
15 KiB
584 lines
15 KiB
// Copyright 2015 The Gogs Authors. All rights reserved. |
|
// Copyright 2018 The Gitea 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" |
|
"io/ioutil" |
|
"net/url" |
|
"path/filepath" |
|
"strings" |
|
|
|
"code.gitea.io/gitea/models" |
|
"code.gitea.io/gitea/modules/auth" |
|
"code.gitea.io/gitea/modules/base" |
|
"code.gitea.io/gitea/modules/context" |
|
"code.gitea.io/gitea/modules/git" |
|
"code.gitea.io/gitea/modules/log" |
|
"code.gitea.io/gitea/modules/markup" |
|
"code.gitea.io/gitea/modules/markup/markdown" |
|
"code.gitea.io/gitea/modules/timeutil" |
|
"code.gitea.io/gitea/modules/util" |
|
) |
|
|
|
const ( |
|
tplWikiStart base.TplName = "repo/wiki/start" |
|
tplWikiView base.TplName = "repo/wiki/view" |
|
tplWikiRevision base.TplName = "repo/wiki/revision" |
|
tplWikiNew base.TplName = "repo/wiki/new" |
|
tplWikiPages base.TplName = "repo/wiki/pages" |
|
) |
|
|
|
// MustEnableWiki check if wiki is enabled, if external then redirect |
|
func MustEnableWiki(ctx *context.Context) { |
|
if !ctx.Repo.CanRead(models.UnitTypeWiki) && |
|
!ctx.Repo.CanRead(models.UnitTypeExternalWiki) { |
|
if log.IsTrace() { |
|
log.Trace("Permission Denied: User %-v cannot read %-v or %-v of repo %-v\n"+ |
|
"User in repo has Permissions: %-+v", |
|
ctx.User, |
|
models.UnitTypeWiki, |
|
models.UnitTypeExternalWiki, |
|
ctx.Repo.Repository, |
|
ctx.Repo.Permission) |
|
} |
|
ctx.NotFound("MustEnableWiki", nil) |
|
return |
|
} |
|
|
|
unit, err := ctx.Repo.Repository.GetUnit(models.UnitTypeExternalWiki) |
|
if err == nil { |
|
ctx.Redirect(unit.ExternalWikiConfig().ExternalWikiURL) |
|
return |
|
} |
|
} |
|
|
|
// PageMeta wiki page meta information |
|
type PageMeta struct { |
|
Name string |
|
SubURL string |
|
UpdatedUnix timeutil.TimeStamp |
|
} |
|
|
|
// findEntryForFile finds the tree entry for a target filepath. |
|
func findEntryForFile(commit *git.Commit, target string) (*git.TreeEntry, error) { |
|
entries, err := commit.ListEntries() |
|
if err != nil { |
|
return nil, err |
|
} |
|
// The longest name should be checked first |
|
for _, entry := range entries { |
|
if entry.IsRegular() && entry.Name() == target { |
|
return entry, nil |
|
} |
|
} |
|
// Then the unescaped, shortest alternative |
|
var unescapedTarget string |
|
if unescapedTarget, err = url.QueryUnescape(target); err != nil { |
|
return nil, err |
|
} |
|
for _, entry := range entries { |
|
if entry.IsRegular() && entry.Name() == unescapedTarget { |
|
return entry, nil |
|
} |
|
} |
|
return nil, nil |
|
} |
|
|
|
func findWikiRepoCommit(ctx *context.Context) (*git.Repository, *git.Commit, error) { |
|
wikiRepo, err := git.OpenRepository(ctx.Repo.Repository.WikiPath()) |
|
if err != nil { |
|
ctx.ServerError("OpenRepository", err) |
|
return nil, nil, err |
|
} |
|
|
|
commit, err := wikiRepo.GetBranchCommit("master") |
|
if err != nil { |
|
return wikiRepo, nil, err |
|
} |
|
return wikiRepo, commit, nil |
|
} |
|
|
|
// wikiContentsByEntry returns the contents of the wiki page referenced by the |
|
// given tree entry. Writes to ctx if an error occurs. |
|
func wikiContentsByEntry(ctx *context.Context, entry *git.TreeEntry) []byte { |
|
reader, err := entry.Blob().DataAsync() |
|
if err != nil { |
|
ctx.ServerError("Blob.Data", err) |
|
return nil |
|
} |
|
defer reader.Close() |
|
content, err := ioutil.ReadAll(reader) |
|
if err != nil { |
|
ctx.ServerError("ReadAll", err) |
|
return nil |
|
} |
|
return content |
|
} |
|
|
|
// wikiContentsByName returns the contents of a wiki page, along with a boolean |
|
// indicating whether the page exists. Writes to ctx if an error occurs. |
|
func wikiContentsByName(ctx *context.Context, commit *git.Commit, wikiName string) ([]byte, *git.TreeEntry, string, bool) { |
|
var entry *git.TreeEntry |
|
var err error |
|
pageFilename := models.WikiNameToFilename(wikiName) |
|
if entry, err = findEntryForFile(commit, pageFilename); err != nil { |
|
ctx.ServerError("findEntryForFile", err) |
|
return nil, nil, "", false |
|
} else if entry == nil { |
|
return nil, nil, "", true |
|
} |
|
return wikiContentsByEntry(ctx, entry), entry, pageFilename, false |
|
} |
|
|
|
func renderViewPage(ctx *context.Context) (*git.Repository, *git.TreeEntry) { |
|
wikiRepo, commit, err := findWikiRepoCommit(ctx) |
|
if err != nil { |
|
if !git.IsErrNotExist(err) { |
|
ctx.ServerError("GetBranchCommit", err) |
|
} |
|
return nil, nil |
|
} |
|
|
|
// Get page list. |
|
entries, err := commit.ListEntries() |
|
if err != nil { |
|
ctx.ServerError("ListEntries", err) |
|
return nil, nil |
|
} |
|
pages := make([]PageMeta, 0, len(entries)) |
|
for _, entry := range entries { |
|
if !entry.IsRegular() { |
|
continue |
|
} |
|
wikiName, err := models.WikiFilenameToName(entry.Name()) |
|
if err != nil { |
|
if models.IsErrWikiInvalidFileName(err) { |
|
continue |
|
} |
|
ctx.ServerError("WikiFilenameToName", err) |
|
return nil, nil |
|
} else if wikiName == "_Sidebar" || wikiName == "_Footer" { |
|
continue |
|
} |
|
pages = append(pages, PageMeta{ |
|
Name: wikiName, |
|
SubURL: models.WikiNameToSubURL(wikiName), |
|
}) |
|
} |
|
ctx.Data["Pages"] = pages |
|
|
|
// get requested pagename |
|
pageName := models.NormalizeWikiName(ctx.Params(":page")) |
|
if len(pageName) == 0 { |
|
pageName = "Home" |
|
} |
|
ctx.Data["PageURL"] = models.WikiNameToSubURL(pageName) |
|
ctx.Data["old_title"] = pageName |
|
ctx.Data["Title"] = pageName |
|
ctx.Data["title"] = pageName |
|
ctx.Data["RequireHighlightJS"] = true |
|
|
|
//lookup filename in wiki - get filecontent, gitTree entry , real filename |
|
data, entry, pageFilename, noEntry := wikiContentsByName(ctx, commit, pageName) |
|
if noEntry { |
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/_pages") |
|
} |
|
if entry == nil || ctx.Written() { |
|
return nil, nil |
|
} |
|
|
|
sidebarContent, _, _, _ := wikiContentsByName(ctx, commit, "_Sidebar") |
|
if ctx.Written() { |
|
return nil, nil |
|
} |
|
|
|
footerContent, _, _, _ := wikiContentsByName(ctx, commit, "_Footer") |
|
if ctx.Written() { |
|
return nil, nil |
|
} |
|
|
|
metas := ctx.Repo.Repository.ComposeMetas() |
|
ctx.Data["content"] = markdown.RenderWiki(data, ctx.Repo.RepoLink, metas) |
|
ctx.Data["sidebarPresent"] = sidebarContent != nil |
|
ctx.Data["sidebarContent"] = markdown.RenderWiki(sidebarContent, ctx.Repo.RepoLink, metas) |
|
ctx.Data["footerPresent"] = footerContent != nil |
|
ctx.Data["footerContent"] = markdown.RenderWiki(footerContent, ctx.Repo.RepoLink, metas) |
|
|
|
// get commit count - wiki revisions |
|
commitsCount, _ := wikiRepo.FileCommitsCount("master", pageFilename) |
|
ctx.Data["CommitCount"] = commitsCount |
|
|
|
return wikiRepo, entry |
|
} |
|
|
|
func renderRevisionPage(ctx *context.Context) (*git.Repository, *git.TreeEntry) { |
|
wikiRepo, commit, err := findWikiRepoCommit(ctx) |
|
if err != nil { |
|
if !git.IsErrNotExist(err) { |
|
ctx.ServerError("GetBranchCommit", err) |
|
} |
|
return nil, nil |
|
} |
|
|
|
// get requested pagename |
|
pageName := models.NormalizeWikiName(ctx.Params(":page")) |
|
if len(pageName) == 0 { |
|
pageName = "Home" |
|
} |
|
ctx.Data["PageURL"] = models.WikiNameToSubURL(pageName) |
|
ctx.Data["old_title"] = pageName |
|
ctx.Data["Title"] = pageName |
|
ctx.Data["title"] = pageName |
|
ctx.Data["RequireHighlightJS"] = true |
|
|
|
//lookup filename in wiki - get filecontent, gitTree entry , real filename |
|
data, entry, pageFilename, noEntry := wikiContentsByName(ctx, commit, pageName) |
|
if noEntry { |
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/_pages") |
|
} |
|
if entry == nil || ctx.Written() { |
|
return nil, nil |
|
} |
|
|
|
ctx.Data["content"] = string(data) |
|
ctx.Data["sidebarPresent"] = false |
|
ctx.Data["sidebarContent"] = "" |
|
ctx.Data["footerPresent"] = false |
|
ctx.Data["footerContent"] = "" |
|
|
|
// get commit count - wiki revisions |
|
commitsCount, _ := wikiRepo.FileCommitsCount("master", pageFilename) |
|
ctx.Data["CommitCount"] = commitsCount |
|
|
|
// get page |
|
page := ctx.QueryInt("page") |
|
if page <= 1 { |
|
page = 1 |
|
} |
|
|
|
// get Commit Count |
|
commitsHistory, err := wikiRepo.CommitsByFileAndRangeNoFollow("master", pageFilename, page) |
|
if err != nil { |
|
ctx.ServerError("CommitsByFileAndRangeNoFollow", err) |
|
return nil, nil |
|
} |
|
commitsHistory = models.ValidateCommitsWithEmails(commitsHistory) |
|
commitsHistory = models.ParseCommitsWithSignature(commitsHistory) |
|
|
|
ctx.Data["Commits"] = commitsHistory |
|
|
|
pager := context.NewPagination(int(commitsCount), git.CommitsRangeSize, page, 5) |
|
pager.SetDefaultParams(ctx) |
|
ctx.Data["Page"] = pager |
|
|
|
return wikiRepo, entry |
|
} |
|
|
|
func renderEditPage(ctx *context.Context) { |
|
_, commit, err := findWikiRepoCommit(ctx) |
|
if err != nil { |
|
if !git.IsErrNotExist(err) { |
|
ctx.ServerError("GetBranchCommit", err) |
|
} |
|
return |
|
} |
|
|
|
// get requested pagename |
|
pageName := models.NormalizeWikiName(ctx.Params(":page")) |
|
if len(pageName) == 0 { |
|
pageName = "Home" |
|
} |
|
ctx.Data["PageURL"] = models.WikiNameToSubURL(pageName) |
|
ctx.Data["old_title"] = pageName |
|
ctx.Data["Title"] = pageName |
|
ctx.Data["title"] = pageName |
|
ctx.Data["RequireHighlightJS"] = true |
|
|
|
//lookup filename in wiki - get filecontent, gitTree entry , real filename |
|
data, entry, _, noEntry := wikiContentsByName(ctx, commit, pageName) |
|
if noEntry { |
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/_pages") |
|
} |
|
if entry == nil || ctx.Written() { |
|
return |
|
} |
|
|
|
ctx.Data["content"] = string(data) |
|
ctx.Data["sidebarPresent"] = false |
|
ctx.Data["sidebarContent"] = "" |
|
ctx.Data["footerPresent"] = false |
|
ctx.Data["footerContent"] = "" |
|
} |
|
|
|
// Wiki renders single wiki page |
|
func Wiki(ctx *context.Context) { |
|
ctx.Data["PageIsWiki"] = true |
|
ctx.Data["CanWriteWiki"] = ctx.Repo.CanWrite(models.UnitTypeWiki) && !ctx.Repo.Repository.IsArchived |
|
|
|
if !ctx.Repo.Repository.HasWiki() { |
|
ctx.Data["Title"] = ctx.Tr("repo.wiki") |
|
ctx.HTML(200, tplWikiStart) |
|
return |
|
} |
|
|
|
wikiRepo, entry := renderViewPage(ctx) |
|
if ctx.Written() { |
|
return |
|
} |
|
if entry == nil { |
|
ctx.Data["Title"] = ctx.Tr("repo.wiki") |
|
ctx.HTML(200, tplWikiStart) |
|
return |
|
} |
|
|
|
wikiPath := entry.Name() |
|
if markup.Type(wikiPath) != markdown.MarkupName { |
|
ext := strings.ToUpper(filepath.Ext(wikiPath)) |
|
ctx.Data["FormatWarning"] = fmt.Sprintf("%s rendering is not supported at the moment. Rendered as Markdown.", ext) |
|
} |
|
// Get last change information. |
|
lastCommit, err := wikiRepo.GetCommitByPath(wikiPath) |
|
if err != nil { |
|
ctx.ServerError("GetCommitByPath", err) |
|
return |
|
} |
|
ctx.Data["Author"] = lastCommit.Author |
|
|
|
ctx.HTML(200, tplWikiView) |
|
} |
|
|
|
// WikiRevision renders file revision list of wiki page |
|
func WikiRevision(ctx *context.Context) { |
|
ctx.Data["PageIsWiki"] = true |
|
ctx.Data["CanWriteWiki"] = ctx.Repo.CanWrite(models.UnitTypeWiki) && !ctx.Repo.Repository.IsArchived |
|
|
|
if !ctx.Repo.Repository.HasWiki() { |
|
ctx.Data["Title"] = ctx.Tr("repo.wiki") |
|
ctx.HTML(200, tplWikiStart) |
|
return |
|
} |
|
|
|
wikiRepo, entry := renderRevisionPage(ctx) |
|
if ctx.Written() { |
|
return |
|
} |
|
if entry == nil { |
|
ctx.Data["Title"] = ctx.Tr("repo.wiki") |
|
ctx.HTML(200, tplWikiStart) |
|
return |
|
} |
|
|
|
// Get last change information. |
|
wikiPath := entry.Name() |
|
lastCommit, err := wikiRepo.GetCommitByPath(wikiPath) |
|
if err != nil { |
|
ctx.ServerError("GetCommitByPath", err) |
|
return |
|
} |
|
ctx.Data["Author"] = lastCommit.Author |
|
|
|
ctx.HTML(200, tplWikiRevision) |
|
} |
|
|
|
// WikiPages render wiki pages list page |
|
func WikiPages(ctx *context.Context) { |
|
if !ctx.Repo.Repository.HasWiki() { |
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki") |
|
return |
|
} |
|
|
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.pages") |
|
ctx.Data["PageIsWiki"] = true |
|
ctx.Data["CanWriteWiki"] = ctx.Repo.CanWrite(models.UnitTypeWiki) && !ctx.Repo.Repository.IsArchived |
|
|
|
wikiRepo, commit, err := findWikiRepoCommit(ctx) |
|
if err != nil { |
|
return |
|
} |
|
|
|
entries, err := commit.ListEntries() |
|
if err != nil { |
|
ctx.ServerError("ListEntries", err) |
|
return |
|
} |
|
pages := make([]PageMeta, 0, len(entries)) |
|
for _, entry := range entries { |
|
if !entry.IsRegular() { |
|
continue |
|
} |
|
c, err := wikiRepo.GetCommitByPath(entry.Name()) |
|
if err != nil { |
|
ctx.ServerError("GetCommit", err) |
|
return |
|
} |
|
wikiName, err := models.WikiFilenameToName(entry.Name()) |
|
if err != nil { |
|
if models.IsErrWikiInvalidFileName(err) { |
|
continue |
|
} |
|
ctx.ServerError("WikiFilenameToName", err) |
|
return |
|
} |
|
pages = append(pages, PageMeta{ |
|
Name: wikiName, |
|
SubURL: models.WikiNameToSubURL(wikiName), |
|
UpdatedUnix: timeutil.TimeStamp(c.Author.When.Unix()), |
|
}) |
|
} |
|
ctx.Data["Pages"] = pages |
|
|
|
ctx.HTML(200, tplWikiPages) |
|
} |
|
|
|
// WikiRaw outputs raw blob requested by user (image for example) |
|
func WikiRaw(ctx *context.Context) { |
|
wikiRepo, commit, err := findWikiRepoCommit(ctx) |
|
if err != nil { |
|
if wikiRepo != nil { |
|
return |
|
} |
|
} |
|
|
|
providedPath := ctx.Params("*") |
|
|
|
var entry *git.TreeEntry |
|
if commit != nil { |
|
// Try to find a file with that name |
|
entry, err = findEntryForFile(commit, providedPath) |
|
if err != nil { |
|
ctx.ServerError("findFile", err) |
|
return |
|
} |
|
|
|
if entry == nil { |
|
// Try to find a wiki page with that name |
|
if strings.HasSuffix(providedPath, ".md") { |
|
providedPath = providedPath[:len(providedPath)-3] |
|
} |
|
|
|
wikiPath := models.WikiNameToFilename(providedPath) |
|
entry, err = findEntryForFile(commit, wikiPath) |
|
if err != nil { |
|
ctx.ServerError("findFile", err) |
|
return |
|
} |
|
} |
|
} |
|
|
|
if entry != nil { |
|
if err = ServeBlob(ctx, entry.Blob()); err != nil { |
|
ctx.ServerError("ServeBlob", err) |
|
} |
|
return |
|
} |
|
|
|
ctx.NotFound("findEntryForFile", nil) |
|
} |
|
|
|
// NewWiki render wiki create page |
|
func NewWiki(ctx *context.Context) { |
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page") |
|
ctx.Data["PageIsWiki"] = true |
|
ctx.Data["RequireSimpleMDE"] = true |
|
|
|
if !ctx.Repo.Repository.HasWiki() { |
|
ctx.Data["title"] = "Home" |
|
} |
|
|
|
ctx.HTML(200, tplWikiNew) |
|
} |
|
|
|
// NewWikiPost response for wiki create request |
|
func NewWikiPost(ctx *context.Context, form auth.NewWikiForm) { |
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page") |
|
ctx.Data["PageIsWiki"] = true |
|
ctx.Data["RequireSimpleMDE"] = true |
|
|
|
if ctx.HasError() { |
|
ctx.HTML(200, tplWikiNew) |
|
return |
|
} |
|
|
|
if util.IsEmptyString(form.Title) { |
|
ctx.RenderWithErr(ctx.Tr("repo.issues.new.title_empty"), tplWikiNew, form) |
|
return |
|
} |
|
|
|
wikiName := models.NormalizeWikiName(form.Title) |
|
if err := ctx.Repo.Repository.AddWikiPage(ctx.User, wikiName, form.Content, form.Message); err != nil { |
|
if models.IsErrWikiReservedName(err) { |
|
ctx.Data["Err_Title"] = true |
|
ctx.RenderWithErr(ctx.Tr("repo.wiki.reserved_page", wikiName), tplWikiNew, &form) |
|
} else if models.IsErrWikiAlreadyExist(err) { |
|
ctx.Data["Err_Title"] = true |
|
ctx.RenderWithErr(ctx.Tr("repo.wiki.page_already_exists"), tplWikiNew, &form) |
|
} else { |
|
ctx.ServerError("AddWikiPage", err) |
|
} |
|
return |
|
} |
|
|
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/" + models.WikiNameToSubURL(wikiName)) |
|
} |
|
|
|
// EditWiki render wiki modify page |
|
func EditWiki(ctx *context.Context) { |
|
ctx.Data["PageIsWiki"] = true |
|
ctx.Data["PageIsWikiEdit"] = true |
|
ctx.Data["RequireSimpleMDE"] = true |
|
|
|
if !ctx.Repo.Repository.HasWiki() { |
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki") |
|
return |
|
} |
|
|
|
renderEditPage(ctx) |
|
if ctx.Written() { |
|
return |
|
} |
|
|
|
ctx.HTML(200, tplWikiNew) |
|
} |
|
|
|
// EditWikiPost response for wiki modify request |
|
func EditWikiPost(ctx *context.Context, form auth.NewWikiForm) { |
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page") |
|
ctx.Data["PageIsWiki"] = true |
|
ctx.Data["RequireSimpleMDE"] = true |
|
|
|
if ctx.HasError() { |
|
ctx.HTML(200, tplWikiNew) |
|
return |
|
} |
|
|
|
oldWikiName := models.NormalizeWikiName(ctx.Params(":page")) |
|
newWikiName := models.NormalizeWikiName(form.Title) |
|
|
|
if err := ctx.Repo.Repository.EditWikiPage(ctx.User, oldWikiName, newWikiName, form.Content, form.Message); err != nil { |
|
ctx.ServerError("EditWikiPage", err) |
|
return |
|
} |
|
|
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/" + models.WikiNameToSubURL(newWikiName)) |
|
} |
|
|
|
// DeleteWikiPagePost delete wiki page |
|
func DeleteWikiPagePost(ctx *context.Context) { |
|
wikiName := models.NormalizeWikiName(ctx.Params(":page")) |
|
if len(wikiName) == 0 { |
|
wikiName = "Home" |
|
} |
|
|
|
if err := ctx.Repo.Repository.DeleteWikiPage(ctx.User, wikiName); err != nil { |
|
ctx.ServerError("DeleteWikiPage", err) |
|
return |
|
} |
|
|
|
ctx.JSON(200, map[string]interface{}{ |
|
"redirect": ctx.Repo.RepoLink + "/wiki/", |
|
}) |
|
}
|
|
|