You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

wiki.go 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. // Copyright 2015 The Gogs Authors. All rights reserved.
  2. // Copyright 2018 The Gitea Authors. All rights reserved.
  3. // Use of this source code is governed by a MIT-style
  4. // license that can be found in the LICENSE file.
  5. package repo
  6. import (
  7. "fmt"
  8. "io/ioutil"
  9. "path/filepath"
  10. "strings"
  11. "code.gitea.io/gitea/models"
  12. "code.gitea.io/gitea/modules/auth"
  13. "code.gitea.io/gitea/modules/base"
  14. "code.gitea.io/gitea/modules/context"
  15. "code.gitea.io/gitea/modules/git"
  16. "code.gitea.io/gitea/modules/log"
  17. "code.gitea.io/gitea/modules/markup"
  18. "code.gitea.io/gitea/modules/markup/markdown"
  19. "code.gitea.io/gitea/modules/util"
  20. )
  21. const (
  22. tplWikiStart base.TplName = "repo/wiki/start"
  23. tplWikiView base.TplName = "repo/wiki/view"
  24. tplWikiNew base.TplName = "repo/wiki/new"
  25. tplWikiPages base.TplName = "repo/wiki/pages"
  26. )
  27. // MustEnableWiki check if wiki is enabled, if external then redirect
  28. func MustEnableWiki(ctx *context.Context) {
  29. if !ctx.Repo.CanRead(models.UnitTypeWiki) &&
  30. !ctx.Repo.CanRead(models.UnitTypeExternalWiki) {
  31. if log.IsTrace() {
  32. log.Trace("Permission Denied: User %-v cannot read %-v or %-v of repo %-v\n"+
  33. "User in repo has Permissions: %-+v",
  34. ctx.User,
  35. models.UnitTypeWiki,
  36. models.UnitTypeExternalWiki,
  37. ctx.Repo.Repository,
  38. ctx.Repo.Permission)
  39. }
  40. ctx.NotFound("MustEnableWiki", nil)
  41. return
  42. }
  43. unit, err := ctx.Repo.Repository.GetUnit(models.UnitTypeExternalWiki)
  44. if err == nil {
  45. ctx.Redirect(unit.ExternalWikiConfig().ExternalWikiURL)
  46. return
  47. }
  48. }
  49. // PageMeta wiki page meat information
  50. type PageMeta struct {
  51. Name string
  52. SubURL string
  53. UpdatedUnix util.TimeStamp
  54. }
  55. // findEntryForFile finds the tree entry for a target filepath.
  56. func findEntryForFile(commit *git.Commit, target string) (*git.TreeEntry, error) {
  57. entries, err := commit.ListEntries()
  58. if err != nil {
  59. return nil, err
  60. }
  61. for _, entry := range entries {
  62. if entry.IsRegular() && entry.Name() == target {
  63. return entry, nil
  64. }
  65. }
  66. return nil, nil
  67. }
  68. func findWikiRepoCommit(ctx *context.Context) (*git.Repository, *git.Commit, error) {
  69. wikiRepo, err := git.OpenRepository(ctx.Repo.Repository.WikiPath())
  70. if err != nil {
  71. ctx.ServerError("OpenRepository", err)
  72. return nil, nil, err
  73. }
  74. commit, err := wikiRepo.GetBranchCommit("master")
  75. if err != nil {
  76. return wikiRepo, nil, err
  77. }
  78. return wikiRepo, commit, nil
  79. }
  80. // wikiContentsByEntry returns the contents of the wiki page referenced by the
  81. // given tree entry. Writes to ctx if an error occurs.
  82. func wikiContentsByEntry(ctx *context.Context, entry *git.TreeEntry) []byte {
  83. reader, err := entry.Blob().DataAsync()
  84. if err != nil {
  85. ctx.ServerError("Blob.Data", err)
  86. return nil
  87. }
  88. defer reader.Close()
  89. content, err := ioutil.ReadAll(reader)
  90. if err != nil {
  91. ctx.ServerError("ReadAll", err)
  92. return nil
  93. }
  94. return content
  95. }
  96. // wikiContentsByName returns the contents of a wiki page, along with a boolean
  97. // indicating whether the page exists. Writes to ctx if an error occurs.
  98. func wikiContentsByName(ctx *context.Context, commit *git.Commit, wikiName string) ([]byte, bool) {
  99. entry, err := findEntryForFile(commit, models.WikiNameToFilename(wikiName))
  100. if err != nil {
  101. ctx.ServerError("findEntryForFile", err)
  102. return nil, false
  103. } else if entry == nil {
  104. return nil, false
  105. }
  106. return wikiContentsByEntry(ctx, entry), true
  107. }
  108. func renderWikiPage(ctx *context.Context, isViewPage bool) (*git.Repository, *git.TreeEntry) {
  109. wikiRepo, commit, err := findWikiRepoCommit(ctx)
  110. if err != nil {
  111. if !git.IsErrNotExist(err) {
  112. ctx.ServerError("GetBranchCommit", err)
  113. }
  114. return nil, nil
  115. }
  116. // Get page list.
  117. if isViewPage {
  118. entries, err := commit.ListEntries()
  119. if err != nil {
  120. ctx.ServerError("ListEntries", err)
  121. return nil, nil
  122. }
  123. pages := make([]PageMeta, 0, len(entries))
  124. for _, entry := range entries {
  125. if !entry.IsRegular() {
  126. continue
  127. }
  128. wikiName, err := models.WikiFilenameToName(entry.Name())
  129. if err != nil {
  130. if models.IsErrWikiInvalidFileName(err) {
  131. continue
  132. }
  133. wikiRepo.Close()
  134. ctx.ServerError("WikiFilenameToName", err)
  135. return nil, nil
  136. } else if wikiName == "_Sidebar" || wikiName == "_Footer" {
  137. continue
  138. }
  139. pages = append(pages, PageMeta{
  140. Name: wikiName,
  141. SubURL: models.WikiNameToSubURL(wikiName),
  142. })
  143. }
  144. ctx.Data["Pages"] = pages
  145. }
  146. pageName := models.NormalizeWikiName(ctx.Params(":page"))
  147. if len(pageName) == 0 {
  148. pageName = "Home"
  149. }
  150. ctx.Data["PageURL"] = models.WikiNameToSubURL(pageName)
  151. ctx.Data["old_title"] = pageName
  152. ctx.Data["Title"] = pageName
  153. ctx.Data["title"] = pageName
  154. ctx.Data["RequireHighlightJS"] = true
  155. pageFilename := models.WikiNameToFilename(pageName)
  156. var entry *git.TreeEntry
  157. if entry, err = findEntryForFile(commit, pageFilename); err != nil {
  158. wikiRepo.Close()
  159. ctx.ServerError("findEntryForFile", err)
  160. return nil, nil
  161. } else if entry == nil {
  162. wikiRepo.Close()
  163. ctx.Redirect(ctx.Repo.RepoLink + "/wiki/_pages")
  164. return nil, nil
  165. }
  166. data := wikiContentsByEntry(ctx, entry)
  167. if ctx.Written() {
  168. if wikiRepo != nil {
  169. wikiRepo.Close()
  170. }
  171. return nil, nil
  172. }
  173. if isViewPage {
  174. sidebarContent, sidebarPresent := wikiContentsByName(ctx, commit, "_Sidebar")
  175. if ctx.Written() {
  176. return nil, nil
  177. }
  178. footerContent, footerPresent := wikiContentsByName(ctx, commit, "_Footer")
  179. if ctx.Written() {
  180. return nil, nil
  181. }
  182. metas := ctx.Repo.Repository.ComposeMetas()
  183. ctx.Data["content"] = markdown.RenderWiki(data, ctx.Repo.RepoLink, metas)
  184. ctx.Data["sidebarPresent"] = sidebarPresent
  185. ctx.Data["sidebarContent"] = markdown.RenderWiki(sidebarContent, ctx.Repo.RepoLink, metas)
  186. ctx.Data["footerPresent"] = footerPresent
  187. ctx.Data["footerContent"] = markdown.RenderWiki(footerContent, ctx.Repo.RepoLink, metas)
  188. } else {
  189. ctx.Data["content"] = string(data)
  190. ctx.Data["sidebarPresent"] = false
  191. ctx.Data["sidebarContent"] = ""
  192. ctx.Data["footerPresent"] = false
  193. ctx.Data["footerContent"] = ""
  194. }
  195. return wikiRepo, entry
  196. }
  197. // Wiki renders single wiki page
  198. func Wiki(ctx *context.Context) {
  199. ctx.Data["PageIsWiki"] = true
  200. ctx.Data["CanWriteWiki"] = ctx.Repo.CanWrite(models.UnitTypeWiki) && !ctx.Repo.Repository.IsArchived
  201. if !ctx.Repo.Repository.HasWiki() {
  202. ctx.Data["Title"] = ctx.Tr("repo.wiki")
  203. ctx.HTML(200, tplWikiStart)
  204. return
  205. }
  206. wikiRepo, entry := renderWikiPage(ctx, true)
  207. if ctx.Written() {
  208. if wikiRepo != nil {
  209. wikiRepo.Close()
  210. }
  211. return
  212. }
  213. defer func() {
  214. if wikiRepo != nil {
  215. wikiRepo.Close()
  216. }
  217. }()
  218. if entry == nil {
  219. ctx.Data["Title"] = ctx.Tr("repo.wiki")
  220. ctx.HTML(200, tplWikiStart)
  221. return
  222. }
  223. wikiPath := entry.Name()
  224. if markup.Type(wikiPath) != markdown.MarkupName {
  225. ext := strings.ToUpper(filepath.Ext(wikiPath))
  226. ctx.Data["FormatWarning"] = fmt.Sprintf("%s rendering is not supported at the moment. Rendered as Markdown.", ext)
  227. }
  228. // Get last change information.
  229. lastCommit, err := wikiRepo.GetCommitByPath(wikiPath)
  230. if err != nil {
  231. ctx.ServerError("GetCommitByPath", err)
  232. return
  233. }
  234. ctx.Data["Author"] = lastCommit.Author
  235. ctx.HTML(200, tplWikiView)
  236. }
  237. // WikiPages render wiki pages list page
  238. func WikiPages(ctx *context.Context) {
  239. if !ctx.Repo.Repository.HasWiki() {
  240. ctx.Redirect(ctx.Repo.RepoLink + "/wiki")
  241. return
  242. }
  243. ctx.Data["Title"] = ctx.Tr("repo.wiki.pages")
  244. ctx.Data["PageIsWiki"] = true
  245. ctx.Data["CanWriteWiki"] = ctx.Repo.CanWrite(models.UnitTypeWiki) && !ctx.Repo.Repository.IsArchived
  246. wikiRepo, commit, err := findWikiRepoCommit(ctx)
  247. if err != nil {
  248. if wikiRepo != nil {
  249. wikiRepo.Close()
  250. }
  251. return
  252. }
  253. entries, err := commit.ListEntries()
  254. if err != nil {
  255. if wikiRepo != nil {
  256. wikiRepo.Close()
  257. }
  258. ctx.ServerError("ListEntries", err)
  259. return
  260. }
  261. pages := make([]PageMeta, 0, len(entries))
  262. for _, entry := range entries {
  263. if !entry.IsRegular() {
  264. continue
  265. }
  266. c, err := wikiRepo.GetCommitByPath(entry.Name())
  267. if err != nil {
  268. if wikiRepo != nil {
  269. wikiRepo.Close()
  270. }
  271. ctx.ServerError("GetCommit", err)
  272. return
  273. }
  274. wikiName, err := models.WikiFilenameToName(entry.Name())
  275. if err != nil {
  276. if models.IsErrWikiInvalidFileName(err) {
  277. continue
  278. }
  279. if wikiRepo != nil {
  280. wikiRepo.Close()
  281. }
  282. ctx.ServerError("WikiFilenameToName", err)
  283. return
  284. }
  285. pages = append(pages, PageMeta{
  286. Name: wikiName,
  287. SubURL: models.WikiNameToSubURL(wikiName),
  288. UpdatedUnix: util.TimeStamp(c.Author.When.Unix()),
  289. })
  290. }
  291. ctx.Data["Pages"] = pages
  292. defer func() {
  293. if wikiRepo != nil {
  294. wikiRepo.Close()
  295. }
  296. }()
  297. ctx.HTML(200, tplWikiPages)
  298. }
  299. // WikiRaw outputs raw blob requested by user (image for example)
  300. func WikiRaw(ctx *context.Context) {
  301. wikiRepo, commit, err := findWikiRepoCommit(ctx)
  302. if err != nil {
  303. if wikiRepo != nil {
  304. return
  305. }
  306. }
  307. defer func() {
  308. wikiRepo.Close()
  309. }()
  310. providedPath := ctx.Params("*")
  311. var entry *git.TreeEntry
  312. if commit != nil {
  313. // Try to find a file with that name
  314. entry, err = findEntryForFile(commit, providedPath)
  315. if err != nil {
  316. ctx.ServerError("findFile", err)
  317. return
  318. }
  319. if entry == nil {
  320. // Try to find a wiki page with that name
  321. if strings.HasSuffix(providedPath, ".md") {
  322. providedPath = providedPath[:len(providedPath)-3]
  323. }
  324. wikiPath := models.WikiNameToFilename(providedPath)
  325. entry, err = findEntryForFile(commit, wikiPath)
  326. if err != nil {
  327. ctx.ServerError("findFile", err)
  328. return
  329. }
  330. }
  331. }
  332. if entry != nil {
  333. if err = ServeBlob(ctx, entry.Blob()); err != nil {
  334. ctx.ServerError("ServeBlob", err)
  335. }
  336. return
  337. }
  338. ctx.NotFound("findEntryForFile", nil)
  339. }
  340. // NewWiki render wiki create page
  341. func NewWiki(ctx *context.Context) {
  342. ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page")
  343. ctx.Data["PageIsWiki"] = true
  344. ctx.Data["RequireSimpleMDE"] = true
  345. if !ctx.Repo.Repository.HasWiki() {
  346. ctx.Data["title"] = "Home"
  347. }
  348. ctx.HTML(200, tplWikiNew)
  349. }
  350. // NewWikiPost response for wiki create request
  351. func NewWikiPost(ctx *context.Context, form auth.NewWikiForm) {
  352. ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page")
  353. ctx.Data["PageIsWiki"] = true
  354. ctx.Data["RequireSimpleMDE"] = true
  355. if ctx.HasError() {
  356. ctx.HTML(200, tplWikiNew)
  357. return
  358. }
  359. if util.IsEmptyString(form.Title) {
  360. ctx.RenderWithErr(ctx.Tr("repo.issues.new.title_empty"), tplWikiNew, form)
  361. return
  362. }
  363. wikiName := models.NormalizeWikiName(form.Title)
  364. if err := ctx.Repo.Repository.AddWikiPage(ctx.User, wikiName, form.Content, form.Message); err != nil {
  365. if models.IsErrWikiReservedName(err) {
  366. ctx.Data["Err_Title"] = true
  367. ctx.RenderWithErr(ctx.Tr("repo.wiki.reserved_page", wikiName), tplWikiNew, &form)
  368. } else if models.IsErrWikiAlreadyExist(err) {
  369. ctx.Data["Err_Title"] = true
  370. ctx.RenderWithErr(ctx.Tr("repo.wiki.page_already_exists"), tplWikiNew, &form)
  371. } else {
  372. ctx.ServerError("AddWikiPage", err)
  373. }
  374. return
  375. }
  376. ctx.Redirect(ctx.Repo.RepoLink + "/wiki/" + models.WikiNameToSubURL(wikiName))
  377. }
  378. // EditWiki render wiki modify page
  379. func EditWiki(ctx *context.Context) {
  380. ctx.Data["PageIsWiki"] = true
  381. ctx.Data["PageIsWikiEdit"] = true
  382. ctx.Data["RequireSimpleMDE"] = true
  383. if !ctx.Repo.Repository.HasWiki() {
  384. ctx.Redirect(ctx.Repo.RepoLink + "/wiki")
  385. return
  386. }
  387. renderWikiPage(ctx, false)
  388. if ctx.Written() {
  389. return
  390. }
  391. ctx.HTML(200, tplWikiNew)
  392. }
  393. // EditWikiPost response for wiki modify request
  394. func EditWikiPost(ctx *context.Context, form auth.NewWikiForm) {
  395. ctx.Data["Title"] = ctx.Tr("repo.wiki.new_page")
  396. ctx.Data["PageIsWiki"] = true
  397. ctx.Data["RequireSimpleMDE"] = true
  398. if ctx.HasError() {
  399. ctx.HTML(200, tplWikiNew)
  400. return
  401. }
  402. oldWikiName := models.NormalizeWikiName(ctx.Params(":page"))
  403. newWikiName := models.NormalizeWikiName(form.Title)
  404. if err := ctx.Repo.Repository.EditWikiPage(ctx.User, oldWikiName, newWikiName, form.Content, form.Message); err != nil {
  405. ctx.ServerError("EditWikiPage", err)
  406. return
  407. }
  408. ctx.Redirect(ctx.Repo.RepoLink + "/wiki/" + models.WikiNameToSubURL(newWikiName))
  409. }
  410. // DeleteWikiPagePost delete wiki page
  411. func DeleteWikiPagePost(ctx *context.Context) {
  412. wikiName := models.NormalizeWikiName(ctx.Params(":page"))
  413. if len(wikiName) == 0 {
  414. wikiName = "Home"
  415. }
  416. if err := ctx.Repo.Repository.DeleteWikiPage(ctx.User, wikiName); err != nil {
  417. ctx.ServerError("DeleteWikiPage", err)
  418. return
  419. }
  420. ctx.JSON(200, map[string]interface{}{
  421. "redirect": ctx.Repo.RepoLink + "/wiki/",
  422. })
  423. }