mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 08:58:24 +00:00 
			
		
		
		
	* Add data-index attribute to issue anchors Signed-off-by: jolheiser <john.olheiser@gmail.com> * Init JS Signed-off-by: jolheiser <john.olheiser@gmail.com> * Add required data to anchor Signed-off-by: jolheiser <john.olheiser@gmail.com> * Finish popup Signed-off-by: jolheiser <john.olheiser@gmail.com> * Revert changes to html.go Signed-off-by: jolheiser <john.olheiser@gmail.com> * Better octicon contexts Signed-off-by: jolheiser <john.olheiser@gmail.com> * Split out popup function for re-use Signed-off-by: jolheiser <john.olheiser@gmail.com> * Style changes, test fixes, and cross-reference support Signed-off-by: jolheiser <john.olheiser@gmail.com> * Prefer em to px Signed-off-by: jolheiser <john.olheiser@gmail.com> * Move label margin to base CSS Signed-off-by: jolheiser <john.olheiser@gmail.com> * Move JS to separate file. Signed-off-by: jolheiser <john.olheiser@gmail.com> * Move JS to features and fix module Signed-off-by: jolheiser <john.olheiser@gmail.com> * Remove query-string and hash Co-Authored-By: silverwind <me@silverwind.io> Co-authored-by: Lauris BH <lauris@nix.lv> Co-authored-by: Antoine GIRARD <sapk@users.noreply.github.com> Co-authored-by: silverwind <me@silverwind.io> Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com> Co-authored-by: zeripath <art27@cantab.net>
		
			
				
	
	
		
			299 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			299 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2017 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 markdown_test
 | |
| 
 | |
| import (
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	. "code.gitea.io/gitea/modules/markup/markdown"
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| const AppURL = "http://localhost:3000/"
 | |
| const Repo = "gogits/gogs"
 | |
| const AppSubURL = AppURL + Repo + "/"
 | |
| 
 | |
| // these values should match the Repo const above
 | |
| var localMetas = map[string]string{
 | |
| 	"user":     "gogits",
 | |
| 	"repo":     "gogs",
 | |
| 	"repoPath": "../../../integrations/gitea-repositories-meta/user13/repo11.git/",
 | |
| }
 | |
| 
 | |
| func TestRender_StandardLinks(t *testing.T) {
 | |
| 	setting.AppURL = AppURL
 | |
| 	setting.AppSubURL = AppSubURL
 | |
| 
 | |
| 	test := func(input, expected, expectedWiki string) {
 | |
| 		buffer := RenderString(input, setting.AppSubURL, nil)
 | |
| 		assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
 | |
| 		bufferWiki := RenderWiki([]byte(input), setting.AppSubURL, nil)
 | |
| 		assert.Equal(t, strings.TrimSpace(expectedWiki), strings.TrimSpace(bufferWiki))
 | |
| 	}
 | |
| 
 | |
| 	googleRendered := `<p><a href="https://google.com/" rel="nofollow">https://google.com/</a></p>`
 | |
| 	test("<https://google.com/>", googleRendered, googleRendered)
 | |
| 
 | |
| 	lnk := util.URLJoin(AppSubURL, "WikiPage")
 | |
| 	lnkWiki := util.URLJoin(AppSubURL, "wiki", "WikiPage")
 | |
| 	test("[WikiPage](WikiPage)",
 | |
| 		`<p><a href="`+lnk+`" rel="nofollow">WikiPage</a></p>`,
 | |
| 		`<p><a href="`+lnkWiki+`" rel="nofollow">WikiPage</a></p>`)
 | |
| }
 | |
| 
 | |
| func TestMisc_IsMarkdownFile(t *testing.T) {
 | |
| 	setting.Markdown.FileExtensions = []string{".md", ".markdown", ".mdown", ".mkd"}
 | |
| 	trueTestCases := []string{
 | |
| 		"test.md",
 | |
| 		"wow.MARKDOWN",
 | |
| 		"LOL.mDoWn",
 | |
| 	}
 | |
| 	falseTestCases := []string{
 | |
| 		"test",
 | |
| 		"abcdefg",
 | |
| 		"abcdefghijklmnopqrstuvwxyz",
 | |
| 		"test.md.test",
 | |
| 	}
 | |
| 
 | |
| 	for _, testCase := range trueTestCases {
 | |
| 		assert.True(t, IsMarkdownFile(testCase))
 | |
| 	}
 | |
| 	for _, testCase := range falseTestCases {
 | |
| 		assert.False(t, IsMarkdownFile(testCase))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestRender_Images(t *testing.T) {
 | |
| 	setting.AppURL = AppURL
 | |
| 	setting.AppSubURL = AppSubURL
 | |
| 
 | |
| 	test := func(input, expected string) {
 | |
| 		buffer := RenderString(input, setting.AppSubURL, nil)
 | |
| 		assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(buffer))
 | |
| 	}
 | |
| 
 | |
| 	url := "../../.images/src/02/train.jpg"
 | |
| 	title := "Train"
 | |
| 	href := "https://gitea.io"
 | |
| 	result := util.URLJoin(AppSubURL, url)
 | |
| 
 | |
| 	test(
 | |
| 		"",
 | |
| 		`<p><a href="`+result+`" rel="nofollow"><img src="`+result+`" alt="`+title+`"/></a></p>`)
 | |
| 
 | |
| 	test(
 | |
| 		"[["+title+"|"+url+"]]",
 | |
| 		`<p><a href="`+result+`" rel="nofollow"><img src="`+result+`" title="`+title+`" alt="`+title+`"/></a></p>`)
 | |
| 	test(
 | |
| 		"[]("+href+")",
 | |
| 		`<p><a href="`+href+`" rel="nofollow"><img src="`+result+`" alt="`+title+`"/></a></p>`)
 | |
| }
 | |
| 
 | |
| func testAnswers(baseURLContent, baseURLImages string) []string {
 | |
| 	return []string{
 | |
| 		`<p>Wiki! Enjoy :)</p>
 | |
| <ul>
 | |
| <li><a href="` + baseURLContent + `/Links" rel="nofollow">Links, Language bindings, Engine bindings</a></li>
 | |
| <li><a href="` + baseURLContent + `/Tips" rel="nofollow">Tips</a></li>
 | |
| </ul>
 | |
| <p>See commit <a href="http://localhost:3000/gogits/gogs/commit/65f1bf27bc" rel="nofollow"><code>65f1bf27bc</code></a></p>
 | |
| <p>Ideas and codes</p>
 | |
| <ul>
 | |
| <li>Bezier widget (by <a href="` + AppURL + `r-lyeh" rel="nofollow">@r-lyeh</a>) <a href="http://localhost:3000/ocornut/imgui/issues/786" class="ref-issue" rel="nofollow">ocornut/imgui#786</a></li>
 | |
| <li>Bezier widget (by <a href="` + AppURL + `r-lyeh" rel="nofollow">@r-lyeh</a>) <a href="http://localhost:3000/gogits/gogs/issues/786" class="ref-issue" rel="nofollow">#786</a></li>
 | |
| <li>Node graph editors <a href="https://github.com/ocornut/imgui/issues/306" rel="nofollow">https://github.com/ocornut/imgui/issues/306</a></li>
 | |
| <li><a href="` + baseURLContent + `/memory_editor_example" rel="nofollow">Memory Editor</a></li>
 | |
| <li><a href="` + baseURLContent + `/plot_var_example" rel="nofollow">Plot var helper</a></li>
 | |
| </ul>
 | |
| `,
 | |
| 		`<h2 id="user-content-what-is-wine-staging">What is Wine Staging?</h2>
 | |
| <p><strong>Wine Staging</strong> on website <a href="http://wine-staging.com" rel="nofollow">wine-staging.com</a>.</p>
 | |
| <h2 id="user-content-quick-links">Quick Links</h2>
 | |
| <p>Here are some links to the most important topics. You can find the full list of pages at the sidebar.</p>
 | |
| <table>
 | |
| <thead>
 | |
| <tr>
 | |
| <th><a href="` + baseURLImages + `/images/icon-install.png" rel="nofollow"><img src="` + baseURLImages + `/images/icon-install.png" title="icon-install.png" alt="images/icon-install.png"/></a></th>
 | |
| <th><a href="` + baseURLContent + `/Installation" rel="nofollow">Installation</a></th>
 | |
| </tr>
 | |
| </thead>
 | |
| <tbody>
 | |
| <tr>
 | |
| <td><a href="` + baseURLImages + `/images/icon-usage.png" rel="nofollow"><img src="` + baseURLImages + `/images/icon-usage.png" title="icon-usage.png" alt="images/icon-usage.png"/></a></td>
 | |
| <td><a href="` + baseURLContent + `/Usage" rel="nofollow">Usage</a></td>
 | |
| </tr>
 | |
| </tbody>
 | |
| </table>
 | |
| `,
 | |
| 		`<p><a href="http://www.excelsiorjet.com/" rel="nofollow">Excelsior JET</a> allows you to create native executables for Windows, Linux and Mac OS X.</p>
 | |
| <ol>
 | |
| <li><a href="https://github.com/libgdx/libgdx/wiki/Gradle-on-the-Commandline#packaging-for-the-desktop" rel="nofollow">Package your libGDX application</a>
 | |
| <a href="` + baseURLImages + `/images/1.png" rel="nofollow"><img src="` + baseURLImages + `/images/1.png" title="1.png" alt="images/1.png"/></a></li>
 | |
| <li>Perform a test run by hitting the Run! button.
 | |
| <a href="` + baseURLImages + `/images/2.png" rel="nofollow"><img src="` + baseURLImages + `/images/2.png" title="2.png" alt="images/2.png"/></a></li>
 | |
| </ol>
 | |
| <h2 id="user-content-custom-id">More tests</h2>
 | |
| <p>(from <a href="https://www.markdownguide.org/extended-syntax/" rel="nofollow">https://www.markdownguide.org/extended-syntax/</a>)</p>
 | |
| <h3 id="user-content-definition-list">Definition list</h3>
 | |
| <dl>
 | |
| <dt>First Term</dt>
 | |
| <dd>This is the definition of the first term.</dd>
 | |
| <dt>Second Term</dt>
 | |
| <dd>This is one definition of the second term.</dd>
 | |
| <dd>This is another definition of the second term.</dd>
 | |
| </dl>
 | |
| <h3 id="user-content-footnotes">Footnotes</h3>
 | |
| <p>Here is a simple footnote,<sup id="fnref:user-content-1"><a href="#fn:user-content-1" rel="nofollow">1</a></sup> and here is a longer one.<sup id="fnref:user-content-bignote"><a href="#fn:user-content-bignote" rel="nofollow">2</a></sup></p>
 | |
| <div>
 | |
| <hr/>
 | |
| <ol>
 | |
| <li id="fn:user-content-1">
 | |
| <p>This is the first footnote. <a href="#fnref:user-content-1" rel="nofollow">↩︎</a></p>
 | |
| </li>
 | |
| <li id="fn:user-content-bignote">
 | |
| <p>Here is one with multiple paragraphs and code.</p>
 | |
| <p>Indent paragraphs to include them in the footnote.</p>
 | |
| <p><code>{ my code }</code></p>
 | |
| <p>Add as many paragraphs as you like. <a href="#fnref:user-content-bignote" rel="nofollow">↩︎</a></p>
 | |
| </li>
 | |
| </ol>
 | |
| </div>
 | |
| `,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Test cases without ambiguous links
 | |
| var sameCases = []string{
 | |
| 	// dear imgui wiki markdown extract: special wiki syntax
 | |
| 	`Wiki! Enjoy :)
 | |
| - [[Links, Language bindings, Engine bindings|Links]]
 | |
| - [[Tips]]
 | |
| 
 | |
| See commit 65f1bf27bc
 | |
| 
 | |
| Ideas and codes
 | |
| 
 | |
| - Bezier widget (by @r-lyeh) ` + AppURL + `ocornut/imgui/issues/786
 | |
| - Bezier widget (by @r-lyeh) ` + AppURL + `gogits/gogs/issues/786
 | |
| - Node graph editors https://github.com/ocornut/imgui/issues/306
 | |
| - [[Memory Editor|memory_editor_example]]
 | |
| - [[Plot var helper|plot_var_example]]`,
 | |
| 	// wine-staging wiki home extract: tables, special wiki syntax, images
 | |
| 	`## What is Wine Staging?
 | |
| **Wine Staging** on website [wine-staging.com](http://wine-staging.com).
 | |
| 
 | |
| ## Quick Links
 | |
| Here are some links to the most important topics. You can find the full list of pages at the sidebar.
 | |
| 
 | |
| | [[images/icon-install.png]]    | [[Installation]]                                         |
 | |
| |--------------------------------|----------------------------------------------------------|
 | |
| | [[images/icon-usage.png]]      | [[Usage]]                                                |
 | |
| `,
 | |
| 	// libgdx wiki page: inline images with special syntax
 | |
| 	`[Excelsior JET](http://www.excelsiorjet.com/) allows you to create native executables for Windows, Linux and Mac OS X.
 | |
| 
 | |
| 1. [Package your libGDX application](https://github.com/libgdx/libgdx/wiki/Gradle-on-the-Commandline#packaging-for-the-desktop)
 | |
| [[images/1.png]]
 | |
| 2. Perform a test run by hitting the Run! button.
 | |
| [[images/2.png]]
 | |
| 
 | |
| ## More tests {#custom-id}
 | |
| 
 | |
| (from https://www.markdownguide.org/extended-syntax/)
 | |
| 
 | |
| ### Definition list
 | |
| 
 | |
| First Term
 | |
| : This is the definition of the first term.
 | |
| 
 | |
| Second Term
 | |
| : This is one definition of the second term.
 | |
| : This is another definition of the second term.
 | |
| 
 | |
| ### Footnotes
 | |
| 
 | |
| Here is a simple footnote,[^1] and here is a longer one.[^bignote]
 | |
| 
 | |
| [^1]: This is the first footnote.
 | |
| 
 | |
| [^bignote]: Here is one with multiple paragraphs and code.
 | |
| 
 | |
|     Indent paragraphs to include them in the footnote.
 | |
| 
 | |
|     ` + "`{ my code }`" + `
 | |
| 
 | |
|     Add as many paragraphs as you like.
 | |
| `,
 | |
| }
 | |
| 
 | |
| func TestTotal_RenderWiki(t *testing.T) {
 | |
| 	answers := testAnswers(util.URLJoin(AppSubURL, "wiki/"), util.URLJoin(AppSubURL, "wiki", "raw/"))
 | |
| 
 | |
| 	for i := 0; i < len(sameCases); i++ {
 | |
| 		line := RenderWiki([]byte(sameCases[i]), AppSubURL, localMetas)
 | |
| 		assert.Equal(t, answers[i], line)
 | |
| 	}
 | |
| 
 | |
| 	testCases := []string{
 | |
| 		// Guard wiki sidebar: special syntax
 | |
| 		`[[Guardfile-DSL / Configuring-Guard|Guardfile-DSL---Configuring-Guard]]`,
 | |
| 		// rendered
 | |
| 		`<p><a href="` + AppSubURL + `wiki/Guardfile-DSL---Configuring-Guard" rel="nofollow">Guardfile-DSL / Configuring-Guard</a></p>
 | |
| `,
 | |
| 		// special syntax
 | |
| 		`[[Name|Link]]`,
 | |
| 		// rendered
 | |
| 		`<p><a href="` + AppSubURL + `wiki/Link" rel="nofollow">Name</a></p>
 | |
| `,
 | |
| 	}
 | |
| 
 | |
| 	for i := 0; i < len(testCases); i += 2 {
 | |
| 		line := RenderWiki([]byte(testCases[i]), AppSubURL, nil)
 | |
| 		assert.Equal(t, testCases[i+1], line)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestTotal_RenderString(t *testing.T) {
 | |
| 	answers := testAnswers(util.URLJoin(AppSubURL, "src", "master/"), util.URLJoin(AppSubURL, "raw", "master/"))
 | |
| 
 | |
| 	for i := 0; i < len(sameCases); i++ {
 | |
| 		line := RenderString(sameCases[i], util.URLJoin(AppSubURL, "src", "master/"), localMetas)
 | |
| 		assert.Equal(t, answers[i], line)
 | |
| 	}
 | |
| 
 | |
| 	testCases := []string{}
 | |
| 
 | |
| 	for i := 0; i < len(testCases); i += 2 {
 | |
| 		line := RenderString(testCases[i], AppSubURL, nil)
 | |
| 		assert.Equal(t, testCases[i+1], line)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestRender_RenderParagraphs(t *testing.T) {
 | |
| 	test := func(t *testing.T, str string, cnt int) {
 | |
| 		unix := []byte(str)
 | |
| 		res := string(RenderRaw(unix, "", false))
 | |
| 		assert.Equal(t, strings.Count(res, "<p"), cnt, "Rendered result for unix should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
 | |
| 
 | |
| 		mac := []byte(strings.ReplaceAll(str, "\n", "\r"))
 | |
| 		res = string(RenderRaw(mac, "", false))
 | |
| 		assert.Equal(t, strings.Count(res, "<p"), cnt, "Rendered result for mac should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
 | |
| 
 | |
| 		dos := []byte(strings.ReplaceAll(str, "\n", "\r\n"))
 | |
| 		res = string(RenderRaw(dos, "", false))
 | |
| 		assert.Equal(t, strings.Count(res, "<p"), cnt, "Rendered result for windows should have %d paragraph(s) but has %d:\n%s\n", cnt, strings.Count(res, "<p"), res)
 | |
| 	}
 | |
| 
 | |
| 	test(t, "\nOne\nTwo\nThree", 1)
 | |
| 	test(t, "\n\nOne\nTwo\nThree", 1)
 | |
| 	test(t, "\n\nOne\nTwo\nThree\n\n\n", 1)
 | |
| 	test(t, "A\n\nB\nC\n", 2)
 | |
| 	test(t, "A\n\n\nB\nC\n", 2)
 | |
| }
 |