2015-11-26 04:10:25 +03:00
// Copyright 2015 The Gogs Authors. All rights reserved.
2018-11-28 14:26:14 +03:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2015-11-26 04:10:25 +03:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package repo
import (
2017-02-14 04:13:59 +03:00
"fmt"
2015-11-27 08:24:24 +03:00
"io/ioutil"
2017-02-14 04:13:59 +03:00
"path/filepath"
2015-11-27 09:50:38 +03:00
"strings"
2015-11-27 08:24:24 +03:00
2016-11-10 19:24:48 +03:00
"code.gitea.io/git"
2015-11-27 08:24:24 +03:00
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/auth"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/context"
2017-04-21 10:01:08 +03:00
"code.gitea.io/gitea/modules/markup"
2017-09-21 08:20:14 +03:00
"code.gitea.io/gitea/modules/markup/markdown"
2017-12-13 16:46:56 +03:00
"code.gitea.io/gitea/modules/util"
2015-11-26 04:10:25 +03:00
)
const (
2016-11-21 13:03:42 +03: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 04:10:25 +03:00
)
2016-11-21 13:03:42 +03:00
// MustEnableWiki check if wiki is enabled, if external then redirect
2016-03-11 19:56:52 +03:00
func MustEnableWiki ( ctx * context . Context ) {
2018-11-28 14:26:14 +03:00
if ! ctx . Repo . CanRead ( models . UnitTypeWiki ) &&
! ctx . Repo . CanRead ( models . UnitTypeExternalWiki ) {
2018-01-11 00:34:17 +03:00
ctx . NotFound ( "MustEnableWiki" , nil )
2015-12-11 12:55:08 +03:00
return
}
2017-02-04 18:53:46 +03:00
unit , err := ctx . Repo . Repository . GetUnit ( models . UnitTypeExternalWiki )
if err == nil {
ctx . Redirect ( unit . ExternalWikiConfig ( ) . ExternalWikiURL )
2015-12-11 12:55:08 +03:00
return
2015-12-05 05:30:33 +03:00
}
}
2016-11-21 13:03:42 +03:00
// PageMeta wiki page meat information
2015-11-27 09:50:38 +03:00
type PageMeta struct {
2017-12-13 16:46:56 +03:00
Name string
SubURL string
UpdatedUnix util . TimeStamp
2015-11-27 09:50:38 +03:00
}
2015-11-26 04:10:25 +03:00
2017-11-28 12:43:51 +03:00
// findEntryForFile finds the tree entry for a target filepath.
func findEntryForFile ( commit * git . Commit , target string ) ( * git . TreeEntry , error ) {
2017-02-14 04:13:59 +03:00
entries , err := commit . ListEntries ( )
if err != nil {
return nil , err
}
2017-11-28 12:43:51 +03:00
for _ , entry := range entries {
if entry . Type == git . ObjectBlob && entry . Name ( ) == target {
return entry , nil
2017-02-14 04:13:59 +03:00
}
}
2017-11-28 12:43:51 +03:00
return nil , nil
2017-02-14 04:13:59 +03:00
}
func findWikiRepoCommit ( ctx * context . Context ) ( * git . Repository , * git . Commit , error ) {
2015-11-27 08:24:24 +03:00
wikiRepo , err := git . OpenRepository ( ctx . Repo . Repository . WikiPath ( ) )
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "OpenRepository" , err )
2017-02-14 04:13:59 +03:00
return nil , nil , err
2015-11-27 08:24:24 +03:00
}
2017-03-20 16:36:19 +03:00
2015-12-10 04:46:05 +03:00
commit , err := wikiRepo . GetBranchCommit ( "master" )
2015-11-27 08:24:24 +03:00
if err != nil {
2017-02-14 04:13:59 +03:00
return wikiRepo , nil , err
}
return wikiRepo , commit , nil
}
2017-11-28 12:43:51 +03:00
// 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 ( ) . Data ( )
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "Blob.Data" , err )
2017-11-28 12:43:51 +03:00
return nil
}
content , err := ioutil . ReadAll ( reader )
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "ReadAll" , err )
2017-11-28 12:43:51 +03:00
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 , bool ) {
entry , err := findEntryForFile ( commit , models . WikiNameToFilename ( wikiName ) )
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "findEntryForFile" , err )
2017-11-28 12:43:51 +03:00
return nil , false
} else if entry == nil {
return nil , false
}
return wikiContentsByEntry ( ctx , entry ) , true
}
2017-02-14 04:13:59 +03:00
func renderWikiPage ( ctx * context . Context , isViewPage bool ) ( * git . Repository , * git . TreeEntry ) {
wikiRepo , commit , err := findWikiRepoCommit ( ctx )
if err != nil {
2018-12-10 01:45:44 +03:00
if ! git . IsErrNotExist ( err ) {
ctx . ServerError ( "GetBranchCommit" , err )
}
2017-02-14 04:13:59 +03:00
return nil , nil
2015-11-27 09:50:38 +03:00
}
// Get page list.
if isViewPage {
entries , err := commit . ListEntries ( )
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "ListEntries" , err )
2017-02-14 04:13:59 +03:00
return nil , nil
2015-11-27 09:50:38 +03:00
}
2017-11-28 12:43:51 +03:00
pages := make ( [ ] PageMeta , 0 , len ( entries ) )
for _ , entry := range entries {
if entry . Type != git . ObjectBlob {
continue
2015-11-27 09:50:38 +03:00
}
2017-11-28 12:43:51 +03:00
wikiName , err := models . WikiFilenameToName ( entry . Name ( ) )
if err != nil {
2018-02-05 17:56:30 +03:00
if models . IsErrWikiInvalidFileName ( err ) {
continue
}
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "WikiFilenameToName" , err )
2017-11-28 12:43:51 +03:00
return nil , nil
} else if wikiName == "_Sidebar" || wikiName == "_Footer" {
continue
}
pages = append ( pages , PageMeta {
Name : wikiName ,
SubURL : models . WikiNameToSubURL ( wikiName ) ,
} )
2015-11-27 09:50:38 +03:00
}
ctx . Data [ "Pages" ] = pages
2015-11-27 08:24:24 +03:00
}
2017-11-28 12:43:51 +03:00
pageName := models . NormalizeWikiName ( ctx . Params ( ":page" ) )
if len ( pageName ) == 0 {
pageName = "Home"
2015-11-27 08:24:24 +03:00
}
2017-11-28 12:43:51 +03:00
ctx . Data [ "PageURL" ] = models . WikiNameToSubURL ( pageName )
2015-11-27 09:50:38 +03:00
ctx . Data [ "old_title" ] = pageName
ctx . Data [ "Title" ] = pageName
ctx . Data [ "title" ] = pageName
2015-11-27 08:24:24 +03:00
ctx . Data [ "RequireHighlightJS" ] = true
2017-11-28 12:43:51 +03:00
pageFilename := models . WikiNameToFilename ( pageName )
2017-02-14 04:13:59 +03:00
var entry * git . TreeEntry
2017-11-28 12:43:51 +03:00
if entry , err = findEntryForFile ( commit , pageFilename ) ; err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "findEntryForFile" , err )
2017-02-14 04:13:59 +03:00
return nil , nil
2017-11-28 12:43:51 +03:00
} else if entry == nil {
2017-02-14 04:13:59 +03:00
ctx . Redirect ( ctx . Repo . RepoLink + "/wiki/_pages" )
return nil , nil
2015-11-27 08:24:24 +03:00
}
2017-11-28 12:43:51 +03:00
data := wikiContentsByEntry ( ctx , entry )
if ctx . Written ( ) {
2017-02-14 04:13:59 +03:00
return nil , nil
}
2017-11-28 12:43:51 +03:00
if isViewPage {
sidebarContent , sidebarPresent := wikiContentsByName ( ctx , commit , "_Sidebar" )
if ctx . Written ( ) {
return nil , nil
2017-02-14 04:13:59 +03:00
}
2017-11-28 12:43:51 +03:00
footerContent , footerPresent := wikiContentsByName ( ctx , commit , "_Footer" )
if ctx . Written ( ) {
return nil , nil
2017-02-14 04:13:59 +03:00
}
2017-11-28 12:43:51 +03:00
2017-02-14 04:13:59 +03:00
metas := ctx . Repo . Repository . ComposeMetas ( )
ctx . Data [ "content" ] = markdown . RenderWiki ( data , ctx . Repo . RepoLink , metas )
ctx . Data [ "sidebarPresent" ] = sidebarPresent
ctx . Data [ "sidebarContent" ] = markdown . RenderWiki ( sidebarContent , ctx . Repo . RepoLink , metas )
ctx . Data [ "footerPresent" ] = footerPresent
ctx . Data [ "footerContent" ] = markdown . RenderWiki ( footerContent , ctx . Repo . RepoLink , metas )
2015-11-27 09:50:38 +03:00
} else {
ctx . Data [ "content" ] = string ( data )
2017-02-14 04:13:59 +03:00
ctx . Data [ "sidebarPresent" ] = false
ctx . Data [ "sidebarContent" ] = ""
ctx . Data [ "footerPresent" ] = false
ctx . Data [ "footerContent" ] = ""
2015-11-27 09:50:38 +03:00
}
2017-02-14 04:13:59 +03:00
return wikiRepo , entry
2015-11-27 09:50:38 +03:00
}
2017-02-14 04:13:59 +03:00
// Wiki renders single wiki page
2016-03-11 19:56:52 +03:00
func Wiki ( ctx * context . Context ) {
2015-11-27 09:50:38 +03:00
ctx . Data [ "PageIsWiki" ] = true
2019-01-23 21:58:38 +03:00
ctx . Data [ "CanWriteWiki" ] = ctx . Repo . CanWrite ( models . UnitTypeWiki ) && ! ctx . Repo . Repository . IsArchived
2015-11-27 09:50:38 +03:00
if ! ctx . Repo . Repository . HasWiki ( ) {
ctx . Data [ "Title" ] = ctx . Tr ( "repo.wiki" )
2016-11-21 13:03:42 +03:00
ctx . HTML ( 200 , tplWikiStart )
2015-11-27 09:50:38 +03:00
return
}
2017-02-14 04:13:59 +03:00
wikiRepo , entry := renderWikiPage ( ctx , true )
2015-11-27 09:50:38 +03:00
if ctx . Written ( ) {
2015-11-27 08:24:24 +03:00
return
}
2017-03-20 16:36:19 +03:00
if entry == nil {
ctx . Data [ "Title" ] = ctx . Tr ( "repo.wiki" )
ctx . HTML ( 200 , tplWikiStart )
return
}
2015-11-27 08:24:24 +03:00
2017-11-28 12:43:51 +03:00
wikiPath := entry . Name ( )
if markup . Type ( wikiPath ) != markdown . MarkupName {
ext := strings . ToUpper ( filepath . Ext ( wikiPath ) )
2017-02-14 04:13:59 +03:00
ctx . Data [ "FormatWarning" ] = fmt . Sprintf ( "%s rendering is not supported at the moment. Rendered as Markdown." , ext )
}
2015-11-27 08:24:24 +03:00
// Get last change information.
2017-11-28 12:43:51 +03:00
lastCommit , err := wikiRepo . GetCommitByPath ( wikiPath )
2015-11-27 08:24:24 +03:00
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "GetCommitByPath" , err )
2015-11-27 08:24:24 +03:00
return
}
ctx . Data [ "Author" ] = lastCommit . Author
2016-11-21 13:03:42 +03:00
ctx . HTML ( 200 , tplWikiView )
2015-11-26 04:10:25 +03:00
}
2016-11-21 13:03:42 +03:00
// WikiPages render wiki pages list page
2016-03-11 19:56:52 +03:00
func WikiPages ( ctx * context . Context ) {
2015-11-27 10:16:12 +03:00
if ! ctx . Repo . Repository . HasWiki ( ) {
ctx . Redirect ( ctx . Repo . RepoLink + "/wiki" )
return
}
2018-11-28 14:26:14 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.wiki.pages" )
ctx . Data [ "PageIsWiki" ] = true
2019-01-23 21:58:38 +03:00
ctx . Data [ "CanWriteWiki" ] = ctx . Repo . CanWrite ( models . UnitTypeWiki ) && ! ctx . Repo . Repository . IsArchived
2018-11-28 14:26:14 +03:00
2017-02-14 04:13:59 +03:00
wikiRepo , commit , err := findWikiRepoCommit ( ctx )
2015-11-27 10:16:12 +03:00
if err != nil {
return
}
entries , err := commit . ListEntries ( )
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "ListEntries" , err )
2015-11-27 10:16:12 +03:00
return
}
pages := make ( [ ] PageMeta , 0 , len ( entries ) )
2017-11-28 12:43:51 +03:00
for _ , entry := range entries {
if entry . Type != git . ObjectBlob {
continue
}
c , err := wikiRepo . GetCommitByPath ( entry . Name ( ) )
if err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "GetCommit" , err )
2017-11-28 12:43:51 +03:00
return
}
wikiName , err := models . WikiFilenameToName ( entry . Name ( ) )
if err != nil {
2018-02-05 17:56:30 +03:00
if models . IsErrWikiInvalidFileName ( err ) {
continue
}
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "WikiFilenameToName" , err )
2017-11-28 12:43:51 +03:00
return
2015-11-27 10:16:12 +03:00
}
2017-11-28 12:43:51 +03:00
pages = append ( pages , PageMeta {
2017-12-13 16:46:56 +03:00
Name : wikiName ,
SubURL : models . WikiNameToSubURL ( wikiName ) ,
UpdatedUnix : util . TimeStamp ( c . Author . When . Unix ( ) ) ,
2017-11-28 12:43:51 +03:00
} )
2015-11-27 10:16:12 +03:00
}
ctx . Data [ "Pages" ] = pages
2015-11-27 08:24:24 +03:00
2016-11-21 13:03:42 +03:00
ctx . HTML ( 200 , tplWikiPages )
2015-11-27 08:24:24 +03:00
}
2017-02-14 04:13:59 +03:00
// 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
}
}
2019-02-06 04:58:55 +03:00
2017-11-28 12:43:51 +03:00
providedPath := ctx . Params ( "*" )
2019-02-06 04:58:55 +03:00
2017-02-14 04:13:59 +03:00
var entry * git . TreeEntry
if commit != nil {
2019-02-06 04:58:55 +03:00
// 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
}
}
2017-02-14 04:13:59 +03:00
}
2019-02-06 04:58:55 +03:00
if entry != nil {
if err = ServeBlob ( ctx , entry . Blob ( ) ) ; err != nil {
ctx . ServerError ( "ServeBlob" , err )
}
2017-11-28 12:43:51 +03:00
return
}
2019-02-06 04:58:55 +03:00
ctx . NotFound ( "findEntryForFile" , nil )
2017-02-14 04:13:59 +03:00
}
2016-11-21 13:03:42 +03:00
// NewWiki render wiki create page
2016-03-11 19:56:52 +03:00
func NewWiki ( ctx * context . Context ) {
2015-11-26 04:10:25 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.wiki.new_page" )
ctx . Data [ "PageIsWiki" ] = true
ctx . Data [ "RequireSimpleMDE" ] = true
2015-11-27 01:33:45 +03:00
if ! ctx . Repo . Repository . HasWiki ( ) {
2015-11-26 04:10:25 +03:00
ctx . Data [ "title" ] = "Home"
}
2016-11-21 13:03:42 +03:00
ctx . HTML ( 200 , tplWikiNew )
2015-11-26 04:10:25 +03:00
}
2017-11-28 12:43:51 +03:00
// NewWikiPost response for wiki create request
2016-03-11 19:56:52 +03:00
func NewWikiPost ( ctx * context . Context , form auth . NewWikiForm ) {
2015-11-27 01:33:45 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.wiki.new_page" )
ctx . Data [ "PageIsWiki" ] = true
ctx . Data [ "RequireSimpleMDE" ] = true
if ctx . HasError ( ) {
2016-11-21 13:03:42 +03:00
ctx . HTML ( 200 , tplWikiNew )
2015-11-27 01:33:45 +03:00
return
}
2019-01-21 14:45:32 +03:00
if util . IsEmptyString ( form . Title ) {
ctx . RenderWithErr ( ctx . Tr ( "repo.issues.new.title_empty" ) , tplWikiNew , form )
return
}
2017-11-28 12:43:51 +03:00
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 ) {
2015-11-27 09:50:38 +03:00
ctx . Data [ "Err_Title" ] = true
2016-11-21 13:03:42 +03:00
ctx . RenderWithErr ( ctx . Tr ( "repo.wiki.page_already_exists" ) , tplWikiNew , & form )
2015-11-27 09:50:38 +03:00
} else {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "AddWikiPage" , err )
2015-11-27 09:50:38 +03:00
}
2015-11-27 01:33:45 +03:00
return
}
2018-05-28 16:38:20 +03:00
ctx . Redirect ( ctx . Repo . RepoLink + "/wiki/" + models . WikiNameToSubURL ( wikiName ) )
2015-11-27 01:33:45 +03:00
}
2016-11-21 13:03:42 +03:00
// EditWiki render wiki modify page
2016-03-11 19:56:52 +03:00
func EditWiki ( ctx * context . Context ) {
2015-11-27 09:50:38 +03: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 13:03:42 +03:00
ctx . HTML ( 200 , tplWikiNew )
2015-11-27 09:50:38 +03:00
}
2017-11-28 12:43:51 +03:00
// EditWikiPost response for wiki modify request
2016-03-11 19:56:52 +03:00
func EditWikiPost ( ctx * context . Context , form auth . NewWikiForm ) {
2015-11-27 09:50:38 +03:00
ctx . Data [ "Title" ] = ctx . Tr ( "repo.wiki.new_page" )
ctx . Data [ "PageIsWiki" ] = true
ctx . Data [ "RequireSimpleMDE" ] = true
if ctx . HasError ( ) {
2016-11-21 13:03:42 +03:00
ctx . HTML ( 200 , tplWikiNew )
2015-11-27 09:50:38 +03:00
return
}
2017-11-28 12:43:51 +03:00
oldWikiName := models . NormalizeWikiName ( ctx . Params ( ":page" ) )
newWikiName := models . NormalizeWikiName ( form . Title )
2017-01-21 15:50:51 +03:00
2017-11-28 12:43:51 +03:00
if err := ctx . Repo . Repository . EditWikiPage ( ctx . User , oldWikiName , newWikiName , form . Content , form . Message ) ; err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "EditWikiPage" , err )
2015-11-27 09:50:38 +03:00
return
}
2018-05-28 16:38:20 +03:00
ctx . Redirect ( ctx . Repo . RepoLink + "/wiki/" + models . WikiNameToSubURL ( newWikiName ) )
2015-11-26 04:10:25 +03:00
}
2016-03-04 01:06:50 +03:00
2016-11-21 13:03:42 +03:00
// DeleteWikiPagePost delete wiki page
2016-03-11 19:56:52 +03:00
func DeleteWikiPagePost ( ctx * context . Context ) {
2017-11-28 12:43:51 +03:00
wikiName := models . NormalizeWikiName ( ctx . Params ( ":page" ) )
if len ( wikiName ) == 0 {
wikiName = "Home"
2016-03-04 01:06:50 +03:00
}
2017-11-28 12:43:51 +03:00
if err := ctx . Repo . Repository . DeleteWikiPage ( ctx . User , wikiName ) ; err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "DeleteWikiPage" , err )
2016-03-04 01:06:50 +03:00
return
}
ctx . JSON ( 200 , map [ string ] interface { } {
"redirect" : ctx . Repo . RepoLink + "/wiki/" ,
} )
}