2015-11-26 09:10:25 +08:00
|
|
|
// Copyright 2015 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 (
|
2015-11-27 13:24:24 +08:00
|
|
|
"io/ioutil"
|
2015-11-27 14:50:38 +08:00
|
|
|
"strings"
|
2015-11-27 15:16:12 +08:00
|
|
|
"time"
|
2015-11-27 13:24:24 +08:00
|
|
|
|
2016-11-11 00:24:48 +08:00
|
|
|
"code.gitea.io/git"
|
2015-11-27 13:24:24 +08:00
|
|
|
|
2016-11-11 00:24:48 +08:00
|
|
|
"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/markdown"
|
2015-11-26 09:10:25 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2016-11-21 18:03:42 +08:00
|
|
|
tplWikiStart base.TplName = "repo/wiki/start"
|
|
|
|
tplWikiView base.TplName = "repo/wiki/view"
|
|
|
|
tplWikiNew base.TplName = "repo/wiki/new"
|
|
|
|
tplWikiPages base.TplName = "repo/wiki/pages"
|
2015-11-26 09:10:25 +08:00
|
|
|
)
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// MustEnableWiki check if wiki is enabled, if external then redirect
|
2016-03-12 00:56:52 +08:00
|
|
|
func MustEnableWiki(ctx *context.Context) {
|
2017-02-04 23:53:46 +08:00
|
|
|
if !ctx.Repo.Repository.EnableUnit(models.UnitTypeWiki) &&
|
|
|
|
!ctx.Repo.Repository.EnableUnit(models.UnitTypeExternalWiki) {
|
2015-12-05 10:30:33 +08:00
|
|
|
ctx.Handle(404, "MustEnableWiki", nil)
|
2015-12-11 17:55:08 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-02-04 23:53:46 +08:00
|
|
|
unit, err := ctx.Repo.Repository.GetUnit(models.UnitTypeExternalWiki)
|
|
|
|
if err == nil {
|
|
|
|
ctx.Redirect(unit.ExternalWikiConfig().ExternalWikiURL)
|
2015-12-11 17:55:08 +08:00
|
|
|
return
|
2015-12-05 10:30:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// PageMeta wiki page meat information
|
2015-11-27 14:50:38 +08:00
|
|
|
type PageMeta struct {
|
2015-11-27 15:16:12 +08:00
|
|
|
Name string
|
|
|
|
URL string
|
|
|
|
Updated time.Time
|
2015-11-27 14:50:38 +08:00
|
|
|
}
|
2015-11-26 09:10:25 +08:00
|
|
|
|
2016-03-12 00:56:52 +08:00
|
|
|
func renderWikiPage(ctx *context.Context, isViewPage bool) (*git.Repository, string) {
|
2015-11-27 13:24:24 +08:00
|
|
|
wikiRepo, err := git.OpenRepository(ctx.Repo.Repository.WikiPath())
|
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "OpenRepository", err)
|
2015-11-27 14:50:38 +08:00
|
|
|
return nil, ""
|
2015-11-27 13:24:24 +08:00
|
|
|
}
|
2015-12-10 09:46:05 +08:00
|
|
|
commit, err := wikiRepo.GetBranchCommit("master")
|
2015-11-27 13:24:24 +08:00
|
|
|
if err != nil {
|
2015-12-10 09:46:05 +08:00
|
|
|
ctx.Handle(500, "GetBranchCommit", err)
|
2015-11-27 14:50:38 +08:00
|
|
|
return nil, ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get page list.
|
|
|
|
if isViewPage {
|
|
|
|
entries, err := commit.ListEntries()
|
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "ListEntries", err)
|
|
|
|
return nil, ""
|
|
|
|
}
|
2015-11-27 15:16:12 +08:00
|
|
|
pages := make([]PageMeta, 0, len(entries))
|
2015-11-27 14:50:38 +08:00
|
|
|
for i := range entries {
|
2016-12-22 17:30:52 +08:00
|
|
|
if entries[i].Type == git.ObjectBlob && strings.HasSuffix(entries[i].Name(), ".md") {
|
2015-11-27 15:16:12 +08:00
|
|
|
name := strings.TrimSuffix(entries[i].Name(), ".md")
|
|
|
|
pages = append(pages, PageMeta{
|
|
|
|
Name: name,
|
|
|
|
URL: models.ToWikiPageURL(name),
|
|
|
|
})
|
2015-11-27 14:50:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ctx.Data["Pages"] = pages
|
2015-11-27 13:24:24 +08:00
|
|
|
}
|
|
|
|
|
2015-11-27 14:50:38 +08:00
|
|
|
pageURL := ctx.Params(":page")
|
|
|
|
if len(pageURL) == 0 {
|
|
|
|
pageURL = "Home"
|
2015-11-27 13:24:24 +08:00
|
|
|
}
|
2015-11-27 14:50:38 +08:00
|
|
|
ctx.Data["PageURL"] = pageURL
|
|
|
|
|
|
|
|
pageName := models.ToWikiPageName(pageURL)
|
|
|
|
ctx.Data["old_title"] = pageName
|
|
|
|
ctx.Data["Title"] = pageName
|
|
|
|
ctx.Data["title"] = pageName
|
2015-11-27 13:24:24 +08:00
|
|
|
ctx.Data["RequireHighlightJS"] = true
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
blob, err := commit.GetBlobByPath(pageURL + ".md")
|
2015-11-27 13:24:24 +08:00
|
|
|
if err != nil {
|
|
|
|
if git.IsErrNotExist(err) {
|
2015-11-27 14:50:38 +08:00
|
|
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/_pages")
|
2015-11-27 13:24:24 +08:00
|
|
|
} else {
|
|
|
|
ctx.Handle(500, "GetBlobByPath", err)
|
|
|
|
}
|
2015-11-27 14:50:38 +08:00
|
|
|
return nil, ""
|
2015-11-27 13:24:24 +08:00
|
|
|
}
|
|
|
|
r, err := blob.Data()
|
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "Data", err)
|
2015-11-27 14:50:38 +08:00
|
|
|
return nil, ""
|
2015-11-27 13:24:24 +08:00
|
|
|
}
|
|
|
|
data, err := ioutil.ReadAll(r)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "ReadAll", err)
|
2015-11-27 14:50:38 +08:00
|
|
|
return nil, ""
|
|
|
|
}
|
|
|
|
if isViewPage {
|
2016-02-21 06:10:05 +08:00
|
|
|
ctx.Data["content"] = string(markdown.Render(data, ctx.Repo.RepoLink, ctx.Repo.Repository.ComposeMetas()))
|
2015-11-27 14:50:38 +08:00
|
|
|
} else {
|
|
|
|
ctx.Data["content"] = string(data)
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
return wikiRepo, pageURL
|
2015-11-27 14:50:38 +08:00
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// Wiki render wiki page
|
2016-03-12 00:56:52 +08:00
|
|
|
func Wiki(ctx *context.Context) {
|
2015-11-27 14:50:38 +08:00
|
|
|
ctx.Data["PageIsWiki"] = true
|
|
|
|
|
|
|
|
if !ctx.Repo.Repository.HasWiki() {
|
|
|
|
ctx.Data["Title"] = ctx.Tr("repo.wiki")
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.HTML(200, tplWikiStart)
|
2015-11-27 14:50:38 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
wikiRepo, pagePath := renderWikiPage(ctx, true)
|
2015-11-27 14:50:38 +08:00
|
|
|
if ctx.Written() {
|
2015-11-27 13:24:24 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get last change information.
|
2017-01-21 20:50:51 +08:00
|
|
|
lastCommit, err := wikiRepo.GetCommitByPath(pagePath + ".md")
|
2015-11-27 13:24:24 +08:00
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "GetCommitByPath", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["Author"] = lastCommit.Author
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.HTML(200, tplWikiView)
|
2015-11-26 09:10:25 +08:00
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// WikiPages render wiki pages list page
|
2016-03-12 00:56:52 +08:00
|
|
|
func WikiPages(ctx *context.Context) {
|
2015-11-27 15:16:12 +08:00
|
|
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.pages")
|
|
|
|
ctx.Data["PageIsWiki"] = true
|
|
|
|
|
|
|
|
if !ctx.Repo.Repository.HasWiki() {
|
|
|
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
wikiRepo, err := git.OpenRepository(ctx.Repo.Repository.WikiPath())
|
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "OpenRepository", err)
|
|
|
|
return
|
|
|
|
}
|
2015-12-10 09:46:05 +08:00
|
|
|
commit, err := wikiRepo.GetBranchCommit("master")
|
2015-11-27 15:16:12 +08:00
|
|
|
if err != nil {
|
2015-12-10 09:46:05 +08:00
|
|
|
ctx.Handle(500, "GetBranchCommit", err)
|
2015-11-27 15:16:12 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
entries, err := commit.ListEntries()
|
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "ListEntries", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pages := make([]PageMeta, 0, len(entries))
|
|
|
|
for i := range entries {
|
2016-12-22 17:30:52 +08:00
|
|
|
if entries[i].Type == git.ObjectBlob && strings.HasSuffix(entries[i].Name(), ".md") {
|
2015-11-27 15:16:12 +08:00
|
|
|
c, err := wikiRepo.GetCommitByPath(entries[i].Name())
|
|
|
|
if err != nil {
|
|
|
|
ctx.Handle(500, "GetCommit", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
name := strings.TrimSuffix(entries[i].Name(), ".md")
|
|
|
|
pages = append(pages, PageMeta{
|
|
|
|
Name: name,
|
|
|
|
URL: models.ToWikiPageURL(name),
|
|
|
|
Updated: c.Author.When,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ctx.Data["Pages"] = pages
|
2015-11-27 13:24:24 +08:00
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.HTML(200, tplWikiPages)
|
2015-11-27 13:24:24 +08:00
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// NewWiki render wiki create page
|
2016-03-12 00:56:52 +08:00
|
|
|
func NewWiki(ctx *context.Context) {
|
2015-11-26 09:10:25 +08:00
|
|
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page")
|
|
|
|
ctx.Data["PageIsWiki"] = true
|
|
|
|
ctx.Data["RequireSimpleMDE"] = true
|
|
|
|
|
2015-11-27 06:33:45 +08:00
|
|
|
if !ctx.Repo.Repository.HasWiki() {
|
2015-11-26 09:10:25 +08:00
|
|
|
ctx.Data["title"] = "Home"
|
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.HTML(200, tplWikiNew)
|
2015-11-26 09:10:25 +08:00
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// NewWikiPost response fro wiki create request
|
2016-03-12 00:56:52 +08:00
|
|
|
func NewWikiPost(ctx *context.Context, form auth.NewWikiForm) {
|
2015-11-27 06:33:45 +08:00
|
|
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page")
|
|
|
|
ctx.Data["PageIsWiki"] = true
|
|
|
|
ctx.Data["RequireSimpleMDE"] = true
|
|
|
|
|
|
|
|
if ctx.HasError() {
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.HTML(200, tplWikiNew)
|
2015-11-27 06:33:45 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
wikiPath := models.ToWikiPageURL(form.Title)
|
|
|
|
|
|
|
|
if err := ctx.Repo.Repository.AddWikiPage(ctx.User, wikiPath, form.Content, form.Message); err != nil {
|
2015-11-27 14:50:38 +08:00
|
|
|
if models.IsErrWikiAlreadyExist(err) {
|
|
|
|
ctx.Data["Err_Title"] = true
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.RenderWithErr(ctx.Tr("repo.wiki.page_already_exists"), tplWikiNew, &form)
|
2015-11-27 14:50:38 +08:00
|
|
|
} else {
|
|
|
|
ctx.Handle(500, "AddWikiPage", err)
|
|
|
|
}
|
2015-11-27 06:33:45 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/" + wikiPath)
|
2015-11-27 06:33:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// EditWiki render wiki modify page
|
2016-03-12 00:56:52 +08:00
|
|
|
func EditWiki(ctx *context.Context) {
|
2015-11-27 14:50:38 +08:00
|
|
|
ctx.Data["PageIsWiki"] = true
|
|
|
|
ctx.Data["PageIsWikiEdit"] = true
|
|
|
|
ctx.Data["RequireSimpleMDE"] = true
|
|
|
|
|
|
|
|
if !ctx.Repo.Repository.HasWiki() {
|
|
|
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
renderWikiPage(ctx, false)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.HTML(200, tplWikiNew)
|
2015-11-27 14:50:38 +08:00
|
|
|
}
|
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// EditWikiPost response fro wiki modify request
|
2016-03-12 00:56:52 +08:00
|
|
|
func EditWikiPost(ctx *context.Context, form auth.NewWikiForm) {
|
2015-11-27 14:50:38 +08:00
|
|
|
ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page")
|
|
|
|
ctx.Data["PageIsWiki"] = true
|
|
|
|
ctx.Data["RequireSimpleMDE"] = true
|
|
|
|
|
|
|
|
if ctx.HasError() {
|
2016-11-21 18:03:42 +08:00
|
|
|
ctx.HTML(200, tplWikiNew)
|
2015-11-27 14:50:38 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
oldWikiPath := ctx.Params(":page")
|
|
|
|
newWikiPath := models.ToWikiPageURL(form.Title)
|
|
|
|
|
|
|
|
if err := ctx.Repo.Repository.EditWikiPage(ctx.User, oldWikiPath, newWikiPath, form.Content, form.Message); err != nil {
|
2015-11-27 14:50:38 +08:00
|
|
|
ctx.Handle(500, "EditWikiPage", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
ctx.Redirect(ctx.Repo.RepoLink + "/wiki/" + newWikiPath)
|
2015-11-26 09:10:25 +08:00
|
|
|
}
|
2016-03-04 06:06:50 +08:00
|
|
|
|
2016-11-21 18:03:42 +08:00
|
|
|
// DeleteWikiPagePost delete wiki page
|
2016-03-12 00:56:52 +08:00
|
|
|
func DeleteWikiPagePost(ctx *context.Context) {
|
2016-03-04 06:06:50 +08:00
|
|
|
pageURL := ctx.Params(":page")
|
|
|
|
if len(pageURL) == 0 {
|
|
|
|
pageURL = "Home"
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:50:51 +08:00
|
|
|
if err := ctx.Repo.Repository.DeleteWikiPage(ctx.User, pageURL); err != nil {
|
2016-03-04 06:06:50 +08:00
|
|
|
ctx.Handle(500, "DeleteWikiPage", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.JSON(200, map[string]interface{}{
|
|
|
|
"redirect": ctx.Repo.RepoLink + "/wiki/",
|
|
|
|
})
|
|
|
|
}
|