mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 08:30:25 +08:00 
			
		
		
		
	Fix URL handling in the whole markdown module, improve test coverage (#1027)
Amended with string to bool change in API SDK. Signed-off-by: Andrew Boyarshin <andrew.boyarshin@gmail.com>
This commit is contained in:
		
				
					committed by
					
						
						Lunny Xiao
					
				
			
			
				
	
			
			
			
						parent
						
							12e71e5706
						
					
				
				
					commit
					0602a44b27
				
			@@ -150,7 +150,7 @@ func composeTplData(subject, body, link string) map[string]interface{} {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
func composeIssueMessage(issue *Issue, doer *User, tplName base.TplName, tos []string, info string) *mailer.Message {
 | 
					func composeIssueMessage(issue *Issue, doer *User, tplName base.TplName, tos []string, info string) *mailer.Message {
 | 
				
			||||||
	subject := issue.mailSubject()
 | 
						subject := issue.mailSubject()
 | 
				
			||||||
	body := string(markdown.RenderSpecialLink([]byte(issue.Content), issue.Repo.HTMLURL(), issue.Repo.ComposeMetas()))
 | 
						body := string(markdown.RenderString(issue.Content, issue.Repo.HTMLURL(), issue.Repo.ComposeMetas()))
 | 
				
			||||||
	data := composeTplData(subject, body, issue.HTMLURL())
 | 
						data := composeTplData(subject, body, issue.HTMLURL())
 | 
				
			||||||
	data["Doer"] = doer
 | 
						data["Doer"] = doer
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -92,10 +92,10 @@ var (
 | 
				
			|||||||
	ShortLinkPattern = regexp.MustCompile(`(\[\[.*\]\]\w*)`)
 | 
						ShortLinkPattern = regexp.MustCompile(`(\[\[.*\]\]\w*)`)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// AnySHA1Pattern allows to split url containing SHA into parts
 | 
						// AnySHA1Pattern allows to split url containing SHA into parts
 | 
				
			||||||
	AnySHA1Pattern = regexp.MustCompile(`http\S+//(\S+)/(\S+)/(\S+)/(\S+)/([0-9a-f]{40})(?:/?([^#\s]+)?(?:#(\S+))?)?`)
 | 
						AnySHA1Pattern = regexp.MustCompile(`(http\S*)://(\S+)/(\S+)/(\S+)/(\S+)/([0-9a-f]{40})(?:/?([^#\s]+)?(?:#(\S+))?)?`)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// IssueFullPattern allows to split issue (and pull) URLs into parts
 | 
						// IssueFullPattern allows to split issue (and pull) URLs into parts
 | 
				
			||||||
	IssueFullPattern = regexp.MustCompile(`(?:^|\s|\()http\S+//((?:[^\s/]+/)+)((?:\w{1,10}-)?[1-9][0-9]*)([\?|#]\S+.(\S+)?)?\b`)
 | 
						IssueFullPattern = regexp.MustCompile(`(?:^|\s|\()(http\S*)://((?:[^\s/]+/)+)((?:\w{1,10}-)?[1-9][0-9]*)([\?|#]\S+.(\S+)?)?\b`)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	validLinksPattern = regexp.MustCompile(`^[a-z][\w-]+://`)
 | 
						validLinksPattern = regexp.MustCompile(`^[a-z][\w-]+://`)
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
@@ -126,10 +126,11 @@ type Renderer struct {
 | 
				
			|||||||
func (r *Renderer) Link(out *bytes.Buffer, link []byte, title []byte, content []byte) {
 | 
					func (r *Renderer) Link(out *bytes.Buffer, link []byte, title []byte, content []byte) {
 | 
				
			||||||
	if len(link) > 0 && !isLink(link) {
 | 
						if len(link) > 0 && !isLink(link) {
 | 
				
			||||||
		if link[0] != '#' {
 | 
							if link[0] != '#' {
 | 
				
			||||||
			mLink := URLJoin(r.urlPrefix, string(link))
 | 
								lnk := string(link)
 | 
				
			||||||
			if r.isWikiMarkdown {
 | 
								if r.isWikiMarkdown {
 | 
				
			||||||
				mLink = URLJoin(r.urlPrefix, "wiki", string(link))
 | 
									lnk = URLJoin("wiki", lnk)
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
 | 
								mLink := URLJoin(r.urlPrefix, lnk)
 | 
				
			||||||
			link = []byte(mLink)
 | 
								link = []byte(mLink)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
@@ -206,12 +207,10 @@ func (r *Renderer) Image(out *bytes.Buffer, link []byte, title []byte, alt []byt
 | 
				
			|||||||
				return
 | 
									return
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
		} else {
 | 
							} else {
 | 
				
			||||||
			if link[0] != '/' {
 | 
								lnk := string(link)
 | 
				
			||||||
				if !strings.HasSuffix(prefix, "/") {
 | 
								lnk = URLJoin(prefix, lnk)
 | 
				
			||||||
					prefix += "/"
 | 
								lnk = strings.Replace(lnk, " ", "+", -1)
 | 
				
			||||||
				}
 | 
								link = []byte(lnk)
 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
			link = []byte(url.QueryEscape(prefix + string(link)))
 | 
					 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -246,10 +245,30 @@ func URLJoin(elem ...string) string {
 | 
				
			|||||||
	last := len(elem) - 1
 | 
						last := len(elem) - 1
 | 
				
			||||||
	for i, item := range elem {
 | 
						for i, item := range elem {
 | 
				
			||||||
		res += item
 | 
							res += item
 | 
				
			||||||
		if !strings.HasSuffix(res, "/") && i != last {
 | 
							if i != last && !strings.HasSuffix(res, "/") {
 | 
				
			||||||
			res += "/"
 | 
								res += "/"
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						cwdIndex := strings.Index(res, "/./")
 | 
				
			||||||
 | 
						for cwdIndex != -1 {
 | 
				
			||||||
 | 
							res = strings.Replace(res, "/./", "/", 1)
 | 
				
			||||||
 | 
							cwdIndex = strings.Index(res, "/./")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						upIndex := strings.Index(res, "/..")
 | 
				
			||||||
 | 
						for upIndex != -1 {
 | 
				
			||||||
 | 
							res = strings.Replace(res, "/..", "", 1)
 | 
				
			||||||
 | 
							prevStart := -1
 | 
				
			||||||
 | 
							for i := upIndex - 1; i >= 0; i-- {
 | 
				
			||||||
 | 
								if res[i] == '/' {
 | 
				
			||||||
 | 
									prevStart = i
 | 
				
			||||||
 | 
									break
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if prevStart != -1 {
 | 
				
			||||||
 | 
								res = res[:prevStart] + res[upIndex:]
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							upIndex = strings.Index(res, "/..")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	return res
 | 
						return res
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -286,6 +305,9 @@ func RenderIssueIndexPattern(rawBytes []byte, urlPrefix string, metas map[string
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// IsSameDomain checks if given url string has the same hostname as current Gitea instance
 | 
					// IsSameDomain checks if given url string has the same hostname as current Gitea instance
 | 
				
			||||||
func IsSameDomain(s string) bool {
 | 
					func IsSameDomain(s string) bool {
 | 
				
			||||||
 | 
						if strings.HasPrefix(s, "/") {
 | 
				
			||||||
 | 
							return true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	if uapp, err := url.Parse(setting.AppURL); err == nil {
 | 
						if uapp, err := url.Parse(setting.AppURL); err == nil {
 | 
				
			||||||
		if u, err := url.Parse(s); err == nil {
 | 
							if u, err := url.Parse(s); err == nil {
 | 
				
			||||||
			return u.Host == uapp.Host
 | 
								return u.Host == uapp.Host
 | 
				
			||||||
@@ -300,26 +322,27 @@ func renderFullSha1Pattern(rawBytes []byte, urlPrefix string) []byte {
 | 
				
			|||||||
	ms := AnySHA1Pattern.FindAllSubmatch(rawBytes, -1)
 | 
						ms := AnySHA1Pattern.FindAllSubmatch(rawBytes, -1)
 | 
				
			||||||
	for _, m := range ms {
 | 
						for _, m := range ms {
 | 
				
			||||||
		all := m[0]
 | 
							all := m[0]
 | 
				
			||||||
		paths := string(m[1])
 | 
							protocol := string(m[1])
 | 
				
			||||||
		var path = "//" + paths
 | 
							paths := string(m[2])
 | 
				
			||||||
		author := string(m[2])
 | 
							path := protocol + "://" + paths
 | 
				
			||||||
		repoName := string(m[3])
 | 
							author := string(m[3])
 | 
				
			||||||
 | 
							repoName := string(m[4])
 | 
				
			||||||
		path = URLJoin(path, author, repoName)
 | 
							path = URLJoin(path, author, repoName)
 | 
				
			||||||
		ltype := "src"
 | 
							ltype := "src"
 | 
				
			||||||
		itemType := m[4]
 | 
							itemType := m[5]
 | 
				
			||||||
		if IsSameDomain(paths) {
 | 
							if IsSameDomain(paths) {
 | 
				
			||||||
			ltype = string(itemType)
 | 
								ltype = string(itemType)
 | 
				
			||||||
		} else if string(itemType) == "commit" {
 | 
							} else if string(itemType) == "commit" {
 | 
				
			||||||
			ltype = "commit"
 | 
								ltype = "commit"
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		sha := m[5]
 | 
							sha := m[6]
 | 
				
			||||||
		var subtree string
 | 
							var subtree string
 | 
				
			||||||
		if len(m) > 6 && len(m[6]) > 0 {
 | 
							if len(m) > 7 && len(m[7]) > 0 {
 | 
				
			||||||
			subtree = string(m[6])
 | 
								subtree = string(m[7])
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		var line []byte
 | 
							var line []byte
 | 
				
			||||||
		if len(m) > 7 && len(m[7]) > 0 {
 | 
							if len(m) > 8 && len(m[8]) > 0 {
 | 
				
			||||||
			line = m[7]
 | 
								line = m[8]
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		urlSuffix := ""
 | 
							urlSuffix := ""
 | 
				
			||||||
		text := base.ShortSha(string(sha))
 | 
							text := base.ShortSha(string(sha))
 | 
				
			||||||
@@ -346,23 +369,18 @@ func renderFullIssuePattern(rawBytes []byte, urlPrefix string) []byte {
 | 
				
			|||||||
	ms := IssueFullPattern.FindAllSubmatch(rawBytes, -1)
 | 
						ms := IssueFullPattern.FindAllSubmatch(rawBytes, -1)
 | 
				
			||||||
	for _, m := range ms {
 | 
						for _, m := range ms {
 | 
				
			||||||
		all := m[0]
 | 
							all := m[0]
 | 
				
			||||||
		paths := bytes.Split(m[1], []byte("/"))
 | 
							protocol := string(m[1])
 | 
				
			||||||
 | 
							paths := bytes.Split(m[2], []byte("/"))
 | 
				
			||||||
		paths = paths[:len(paths)-1]
 | 
							paths = paths[:len(paths)-1]
 | 
				
			||||||
		if bytes.HasPrefix(paths[0], []byte("gist.")) {
 | 
							if bytes.HasPrefix(paths[0], []byte("gist.")) {
 | 
				
			||||||
			continue
 | 
								continue
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		var path string
 | 
							path := protocol + "://" + string(m[2])
 | 
				
			||||||
		if len(paths) > 3 {
 | 
							id := string(m[3])
 | 
				
			||||||
			// Internal one
 | 
					 | 
				
			||||||
			path = URLJoin(urlPrefix, "issues")
 | 
					 | 
				
			||||||
		} else {
 | 
					 | 
				
			||||||
			path = "//" + string(m[1])
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		id := string(m[2])
 | 
					 | 
				
			||||||
		path = URLJoin(path, id)
 | 
							path = URLJoin(path, id)
 | 
				
			||||||
		var comment []byte
 | 
							var comment []byte
 | 
				
			||||||
		if len(m) > 3 {
 | 
							if len(m) > 3 {
 | 
				
			||||||
			comment = m[3]
 | 
								comment = m[4]
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		urlSuffix := ""
 | 
							urlSuffix := ""
 | 
				
			||||||
		text := "#" + id
 | 
							text := "#" + id
 | 
				
			||||||
@@ -394,8 +412,13 @@ func lastIndexOfByte(sl []byte, target byte) int {
 | 
				
			|||||||
	return -1
 | 
						return -1
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// renderShortLinks processes [[syntax]]
 | 
					// RenderShortLinks processes [[syntax]]
 | 
				
			||||||
func renderShortLinks(rawBytes []byte, urlPrefix string, noLink bool) []byte {
 | 
					//
 | 
				
			||||||
 | 
					// noLink flag disables making link tags when set to true
 | 
				
			||||||
 | 
					// so this function just replaces the whole [[...]] with the content text
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// isWikiMarkdown is a flag to choose linking url prefix
 | 
				
			||||||
 | 
					func RenderShortLinks(rawBytes []byte, urlPrefix string, noLink bool, isWikiMarkdown bool) []byte {
 | 
				
			||||||
	ms := ShortLinkPattern.FindAll(rawBytes, -1)
 | 
						ms := ShortLinkPattern.FindAll(rawBytes, -1)
 | 
				
			||||||
	for _, m := range ms {
 | 
						for _, m := range ms {
 | 
				
			||||||
		orig := bytes.TrimSpace(m)
 | 
							orig := bytes.TrimSpace(m)
 | 
				
			||||||
@@ -482,11 +505,17 @@ func renderShortLinks(rawBytes []byte, urlPrefix string, noLink bool) []byte {
 | 
				
			|||||||
		}
 | 
							}
 | 
				
			||||||
		absoluteLink := isLink([]byte(link))
 | 
							absoluteLink := isLink([]byte(link))
 | 
				
			||||||
		if !absoluteLink {
 | 
							if !absoluteLink {
 | 
				
			||||||
			link = url.QueryEscape(link)
 | 
								link = strings.Replace(link, " ", "+", -1)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		if image {
 | 
							if image {
 | 
				
			||||||
			if !absoluteLink {
 | 
								if !absoluteLink {
 | 
				
			||||||
				link = URLJoin(urlPrefix, "wiki", "raw", link)
 | 
									if IsSameDomain(urlPrefix) {
 | 
				
			||||||
 | 
										urlPrefix = strings.Replace(urlPrefix, "/src/", "/raw/", 1)
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									if isWikiMarkdown {
 | 
				
			||||||
 | 
										link = URLJoin("wiki", "raw", link)
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									link = URLJoin(urlPrefix, link)
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			title := props["title"]
 | 
								title := props["title"]
 | 
				
			||||||
			if title == "" {
 | 
								if title == "" {
 | 
				
			||||||
@@ -504,7 +533,10 @@ func renderShortLinks(rawBytes []byte, urlPrefix string, noLink bool) []byte {
 | 
				
			|||||||
			}
 | 
								}
 | 
				
			||||||
			name = fmt.Sprintf(`<img src="%s" %s title="%s" />`, link, alt, title)
 | 
								name = fmt.Sprintf(`<img src="%s" %s title="%s" />`, link, alt, title)
 | 
				
			||||||
		} else if !absoluteLink {
 | 
							} else if !absoluteLink {
 | 
				
			||||||
			link = URLJoin(urlPrefix, "wiki", link)
 | 
								if isWikiMarkdown {
 | 
				
			||||||
 | 
									link = URLJoin("wiki", link)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								link = URLJoin(urlPrefix, link)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		if noLink {
 | 
							if noLink {
 | 
				
			||||||
			rawBytes = bytes.Replace(rawBytes, orig, []byte(name), -1)
 | 
								rawBytes = bytes.Replace(rawBytes, orig, []byte(name), -1)
 | 
				
			||||||
@@ -527,7 +559,7 @@ func RenderCrossReferenceIssueIndexPattern(rawBytes []byte, urlPrefix string, me
 | 
				
			|||||||
		repo := string(bytes.Split(m, []byte("#"))[0])
 | 
							repo := string(bytes.Split(m, []byte("#"))[0])
 | 
				
			||||||
		issue := string(bytes.Split(m, []byte("#"))[1])
 | 
							issue := string(bytes.Split(m, []byte("#"))[1])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		link := fmt.Sprintf(`<a href="%s">%s</a>`, URLJoin(urlPrefix, repo, "issues", issue), m)
 | 
							link := fmt.Sprintf(`<a href="%s">%s</a>`, URLJoin(setting.AppURL, repo, "issues", issue), m)
 | 
				
			||||||
		rawBytes = bytes.Replace(rawBytes, m, []byte(link), 1)
 | 
							rawBytes = bytes.Replace(rawBytes, m, []byte(link), 1)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return rawBytes
 | 
						return rawBytes
 | 
				
			||||||
@@ -548,7 +580,7 @@ func renderSha1CurrentPattern(rawBytes []byte, urlPrefix string) []byte {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// RenderSpecialLink renders mentions, indexes and SHA1 strings to corresponding links.
 | 
					// RenderSpecialLink renders mentions, indexes and SHA1 strings to corresponding links.
 | 
				
			||||||
func RenderSpecialLink(rawBytes []byte, urlPrefix string, metas map[string]string) []byte {
 | 
					func RenderSpecialLink(rawBytes []byte, urlPrefix string, metas map[string]string, isWikiMarkdown bool) []byte {
 | 
				
			||||||
	ms := MentionPattern.FindAll(rawBytes, -1)
 | 
						ms := MentionPattern.FindAll(rawBytes, -1)
 | 
				
			||||||
	for _, m := range ms {
 | 
						for _, m := range ms {
 | 
				
			||||||
		m = m[bytes.Index(m, []byte("@")):]
 | 
							m = m[bytes.Index(m, []byte("@")):]
 | 
				
			||||||
@@ -556,7 +588,7 @@ func RenderSpecialLink(rawBytes []byte, urlPrefix string, metas map[string]strin
 | 
				
			|||||||
			[]byte(fmt.Sprintf(`<a href="%s">%s</a>`, URLJoin(setting.AppURL, string(m[1:])), m)), -1)
 | 
								[]byte(fmt.Sprintf(`<a href="%s">%s</a>`, URLJoin(setting.AppURL, string(m[1:])), m)), -1)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	rawBytes = renderShortLinks(rawBytes, urlPrefix, false)
 | 
						rawBytes = RenderShortLinks(rawBytes, urlPrefix, false, isWikiMarkdown)
 | 
				
			||||||
	rawBytes = RenderIssueIndexPattern(rawBytes, urlPrefix, metas)
 | 
						rawBytes = RenderIssueIndexPattern(rawBytes, urlPrefix, metas)
 | 
				
			||||||
	rawBytes = RenderCrossReferenceIssueIndexPattern(rawBytes, urlPrefix, metas)
 | 
						rawBytes = RenderCrossReferenceIssueIndexPattern(rawBytes, urlPrefix, metas)
 | 
				
			||||||
	rawBytes = renderFullSha1Pattern(rawBytes, urlPrefix)
 | 
						rawBytes = renderFullSha1Pattern(rawBytes, urlPrefix)
 | 
				
			||||||
@@ -601,7 +633,7 @@ var noEndTags = []string{"img", "input", "br", "hr"}
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// PostProcess treats different types of HTML differently,
 | 
					// PostProcess treats different types of HTML differently,
 | 
				
			||||||
// and only renders special links for plain text blocks.
 | 
					// and only renders special links for plain text blocks.
 | 
				
			||||||
func PostProcess(rawHTML []byte, urlPrefix string, metas map[string]string) []byte {
 | 
					func PostProcess(rawHTML []byte, urlPrefix string, metas map[string]string, isWikiMarkdown bool) []byte {
 | 
				
			||||||
	startTags := make([]string, 0, 5)
 | 
						startTags := make([]string, 0, 5)
 | 
				
			||||||
	var buf bytes.Buffer
 | 
						var buf bytes.Buffer
 | 
				
			||||||
	tokenizer := html.NewTokenizer(bytes.NewReader(rawHTML))
 | 
						tokenizer := html.NewTokenizer(bytes.NewReader(rawHTML))
 | 
				
			||||||
@@ -611,7 +643,7 @@ OUTER_LOOP:
 | 
				
			|||||||
		token := tokenizer.Token()
 | 
							token := tokenizer.Token()
 | 
				
			||||||
		switch token.Type {
 | 
							switch token.Type {
 | 
				
			||||||
		case html.TextToken:
 | 
							case html.TextToken:
 | 
				
			||||||
			buf.Write(RenderSpecialLink([]byte(token.String()), urlPrefix, metas))
 | 
								buf.Write(RenderSpecialLink([]byte(token.String()), urlPrefix, metas, isWikiMarkdown))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		case html.StartTagToken:
 | 
							case html.StartTagToken:
 | 
				
			||||||
			buf.WriteString(token.String())
 | 
								buf.WriteString(token.String())
 | 
				
			||||||
@@ -623,7 +655,7 @@ OUTER_LOOP:
 | 
				
			|||||||
					token = tokenizer.Token()
 | 
										token = tokenizer.Token()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
					// Copy the token to the output verbatim
 | 
										// Copy the token to the output verbatim
 | 
				
			||||||
					buf.Write(renderShortLinks([]byte(token.String()), urlPrefix, true))
 | 
										buf.Write(RenderShortLinks([]byte(token.String()), urlPrefix, true, isWikiMarkdown))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
					if token.Type == html.StartTagToken {
 | 
										if token.Type == html.StartTagToken {
 | 
				
			||||||
						if !com.IsSliceContainsStr(noEndTags, token.Data) {
 | 
											if !com.IsSliceContainsStr(noEndTags, token.Data) {
 | 
				
			||||||
@@ -673,9 +705,9 @@ OUTER_LOOP:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// Render renders Markdown to HTML with all specific handling stuff.
 | 
					// Render renders Markdown to HTML with all specific handling stuff.
 | 
				
			||||||
func render(rawBytes []byte, urlPrefix string, metas map[string]string, isWikiMarkdown bool) []byte {
 | 
					func render(rawBytes []byte, urlPrefix string, metas map[string]string, isWikiMarkdown bool) []byte {
 | 
				
			||||||
	urlPrefix = strings.Replace(urlPrefix, " ", "%20", -1)
 | 
						urlPrefix = strings.Replace(urlPrefix, " ", "+", -1)
 | 
				
			||||||
	result := RenderRaw(rawBytes, urlPrefix, isWikiMarkdown)
 | 
						result := RenderRaw(rawBytes, urlPrefix, isWikiMarkdown)
 | 
				
			||||||
	result = PostProcess(result, urlPrefix, metas)
 | 
						result = PostProcess(result, urlPrefix, metas, isWikiMarkdown)
 | 
				
			||||||
	result = Sanitizer.SanitizeBytes(result)
 | 
						result = Sanitizer.SanitizeBytes(result)
 | 
				
			||||||
	return result
 | 
						return result
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -55,7 +55,7 @@ func testRenderIssueIndexPattern(t *testing.T, input, expected string, metas map
 | 
				
			|||||||
		string(RenderIssueIndexPattern([]byte(input), AppSubURL, metas)))
 | 
							string(RenderIssueIndexPattern([]byte(input), AppSubURL, metas)))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestRenderIssueIndexPattern(t *testing.T) {
 | 
					func TestRender_IssueIndexPattern(t *testing.T) {
 | 
				
			||||||
	// numeric: render inputs without valid mentions
 | 
						// numeric: render inputs without valid mentions
 | 
				
			||||||
	test := func(s string) {
 | 
						test := func(s string) {
 | 
				
			||||||
		testRenderIssueIndexPattern(t, s, s, nil)
 | 
							testRenderIssueIndexPattern(t, s, s, nil)
 | 
				
			||||||
@@ -82,7 +82,7 @@ func TestRenderIssueIndexPattern(t *testing.T) {
 | 
				
			|||||||
	test("test #54321issue")
 | 
						test("test #54321issue")
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestRenderIssueIndexPattern2(t *testing.T) {
 | 
					func TestRender_IssueIndexPattern2(t *testing.T) {
 | 
				
			||||||
	setting.AppURL = AppURL
 | 
						setting.AppURL = AppURL
 | 
				
			||||||
	setting.AppSubURL = AppSubURL
 | 
						setting.AppSubURL = AppSubURL
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -119,7 +119,7 @@ func TestRenderIssueIndexPattern2(t *testing.T) {
 | 
				
			|||||||
	test("#1 (#4321) test", "%s (%s) test", 1, 4321)
 | 
						test("#1 (#4321) test", "%s (%s) test", 1, 4321)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestRenderIssueIndexPattern3(t *testing.T) {
 | 
					func TestRender_IssueIndexPattern3(t *testing.T) {
 | 
				
			||||||
	setting.AppURL = AppURL
 | 
						setting.AppURL = AppURL
 | 
				
			||||||
	setting.AppSubURL = AppSubURL
 | 
						setting.AppSubURL = AppSubURL
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -146,7 +146,7 @@ func TestRenderIssueIndexPattern3(t *testing.T) {
 | 
				
			|||||||
	test("ABC-0123")         // no leading zero
 | 
						test("ABC-0123")         // no leading zero
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestRenderIssueIndexPattern4(t *testing.T) {
 | 
					func TestRender_IssueIndexPattern4(t *testing.T) {
 | 
				
			||||||
	setting.AppURL = AppURL
 | 
						setting.AppURL = AppURL
 | 
				
			||||||
	setting.AppSubURL = AppSubURL
 | 
						setting.AppSubURL = AppSubURL
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -164,15 +164,15 @@ func TestRenderIssueIndexPattern4(t *testing.T) {
 | 
				
			|||||||
	test("test issue ABCDEFGHIJ-1234567890", "test issue %s", "ABCDEFGHIJ-1234567890")
 | 
						test("test issue ABCDEFGHIJ-1234567890", "test issue %s", "ABCDEFGHIJ-1234567890")
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestRenderer_AutoLink(t *testing.T) {
 | 
					func TestRender_AutoLink(t *testing.T) {
 | 
				
			||||||
	setting.AppURL = AppURL
 | 
						setting.AppURL = AppURL
 | 
				
			||||||
	setting.AppSubURL = AppSubURL
 | 
						setting.AppSubURL = AppSubURL
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	SubURLNoProtocol := setting.AppSubURL[5:]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	test := func(input, expected string) {
 | 
						test := func(input, expected string) {
 | 
				
			||||||
		buffer := RenderSpecialLink([]byte(input), setting.AppSubURL, map[string]string{})
 | 
							buffer := RenderSpecialLink([]byte(input), setting.AppSubURL, nil, false)
 | 
				
			||||||
		assert.Equal(t, expected, string(buffer))
 | 
							assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
 | 
				
			||||||
 | 
							buffer = RenderSpecialLink([]byte(input), setting.AppSubURL, nil, true)
 | 
				
			||||||
 | 
							assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// render valid issue URLs
 | 
						// render valid issue URLs
 | 
				
			||||||
@@ -180,54 +180,98 @@ func TestRenderer_AutoLink(t *testing.T) {
 | 
				
			|||||||
		numericIssueLink(URLJoin(setting.AppSubURL, "issues"), 3333))
 | 
							numericIssueLink(URLJoin(setting.AppSubURL, "issues"), 3333))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// render external issue URLs
 | 
						// render external issue URLs
 | 
				
			||||||
	tmp := "//1111/2222/ssss-issues/3333?param=blah&blahh=333"
 | 
						tmp := "http://1111/2222/ssss-issues/3333?param=blah&blahh=333"
 | 
				
			||||||
	test("http:"+tmp,
 | 
						test(tmp, "<a href=\""+tmp+"\">#3333 <i class='comment icon'></i></a>")
 | 
				
			||||||
		"<a href=\""+tmp+"\">#3333 <i class='comment icon'></i></a>")
 | 
						test("http://test.com/issues/33333", numericIssueLink("http://test.com/issues", 33333))
 | 
				
			||||||
	test("http://test.com/issues/33333", numericIssueLink("//test.com/issues", 33333))
 | 
						test("https://issues/333", numericIssueLink("https://issues", 333))
 | 
				
			||||||
	test("https://issues/333", numericIssueLink("//issues", 333))
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// render valid commit URLs
 | 
						// render valid commit URLs
 | 
				
			||||||
	tmp = URLJoin(SubURLNoProtocol, "commit", "d8a994ef243349f321568f9e36d5c3f444b99cae")
 | 
						tmp = URLJoin(AppSubURL, "commit", "d8a994ef243349f321568f9e36d5c3f444b99cae")
 | 
				
			||||||
	test("http://"+tmp, "<a href=\""+tmp+"\">d8a994ef24</a>")
 | 
						test(tmp, "<a href=\""+tmp+"\">d8a994ef24</a>")
 | 
				
			||||||
	tmp += "#diff-2"
 | 
						tmp += "#diff-2"
 | 
				
			||||||
	test("http://"+tmp, "<a href=\""+tmp+"\">d8a994ef24 (diff-2)</a>")
 | 
						test(tmp, "<a href=\""+tmp+"\">d8a994ef24 (diff-2)</a>")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// render other commit URLs
 | 
						// render other commit URLs
 | 
				
			||||||
	tmp = "//external-link.gogs.io/gogs/gogs/commit/d8a994ef243349f321568f9e36d5c3f444b99cae#diff-2"
 | 
						tmp = "https://external-link.gogs.io/gogs/gogs/commit/d8a994ef243349f321568f9e36d5c3f444b99cae#diff-2"
 | 
				
			||||||
	test("https:"+tmp, "<a href=\""+tmp+"\">d8a994ef24 (diff-2)</a>")
 | 
						test(tmp, "<a href=\""+tmp+"\">d8a994ef24 (diff-2)</a>")
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					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(string(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 := URLJoin(AppSubURL, "WikiPage")
 | 
				
			||||||
 | 
						lnkWiki := 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 TestRender_ShortLinks(t *testing.T) {
 | 
					func TestRender_ShortLinks(t *testing.T) {
 | 
				
			||||||
	setting.AppURL = AppURL
 | 
						setting.AppURL = AppURL
 | 
				
			||||||
	setting.AppSubURL = AppSubURL
 | 
						setting.AppSubURL = AppSubURL
 | 
				
			||||||
 | 
						tree := URLJoin(AppSubURL, "src", "master")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	test := func(input, expected string) {
 | 
						test := func(input, expected, expectedWiki string) {
 | 
				
			||||||
		buffer := RenderString(input, setting.AppSubURL, nil)
 | 
							buffer := RenderString(input, tree, nil)
 | 
				
			||||||
		assert.Equal(t, expected, string(buffer))
 | 
							assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
 | 
				
			||||||
 | 
							buffer = RenderWiki([]byte(input), setting.AppSubURL, nil)
 | 
				
			||||||
 | 
							assert.Equal(t, strings.TrimSpace(expectedWiki), strings.TrimSpace(string(buffer)))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var url = URLJoin(AppSubURL, "wiki", "Link")
 | 
						rawtree := URLJoin(AppSubURL, "raw", "master")
 | 
				
			||||||
	var imgurl = URLJoin(AppSubURL, "wiki", "raw", "Link.jpg")
 | 
						url := URLJoin(tree, "Link")
 | 
				
			||||||
	var favicon = "http://google.com/favicon.ico"
 | 
						imgurl := URLJoin(rawtree, "Link.jpg")
 | 
				
			||||||
 | 
						urlWiki := URLJoin(AppSubURL, "wiki", "Link")
 | 
				
			||||||
 | 
						imgurlWiki := URLJoin(AppSubURL, "wiki", "raw", "Link.jpg")
 | 
				
			||||||
 | 
						favicon := "http://google.com/favicon.ico"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	test("[[Link]]", `<p><a href="`+url+`" rel="nofollow">Link</a></p>
 | 
						test(
 | 
				
			||||||
`)
 | 
							"[[Link]]",
 | 
				
			||||||
	test("[[Link.jpg]]", `<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="Link.jpg" title="Link.jpg"/></a></p>
 | 
							`<p><a href="`+url+`" rel="nofollow">Link</a></p>`,
 | 
				
			||||||
`)
 | 
							`<p><a href="`+urlWiki+`" rel="nofollow">Link</a></p>`)
 | 
				
			||||||
	test("[["+favicon+"]]", `<p><a href="`+favicon+`" rel="nofollow"><img src="`+favicon+`" title="favicon.ico"/></a></p>
 | 
						test(
 | 
				
			||||||
`)
 | 
							"[[Link.jpg]]",
 | 
				
			||||||
	test("[[Name|Link]]", `<p><a href="`+url+`" rel="nofollow">Name</a></p>
 | 
							`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="Link.jpg" title="Link.jpg"/></a></p>`,
 | 
				
			||||||
`)
 | 
							`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" alt="Link.jpg" title="Link.jpg"/></a></p>`)
 | 
				
			||||||
	test("[[Name|Link.jpg]]", `<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="Name" title="Name"/></a></p>
 | 
						test(
 | 
				
			||||||
`)
 | 
							"[["+favicon+"]]",
 | 
				
			||||||
	test("[[Name|Link.jpg|alt=AltName]]", `<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="AltName" title="AltName"/></a></p>
 | 
							`<p><a href="`+favicon+`" rel="nofollow"><img src="`+favicon+`" title="favicon.ico"/></a></p>`,
 | 
				
			||||||
`)
 | 
							`<p><a href="`+favicon+`" rel="nofollow"><img src="`+favicon+`" title="favicon.ico"/></a></p>`)
 | 
				
			||||||
	test("[[Name|Link.jpg|title=Title]]", `<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="Title" title="Title"/></a></p>
 | 
						test(
 | 
				
			||||||
`)
 | 
							"[[Name|Link]]",
 | 
				
			||||||
	test("[[Name|Link.jpg|alt=AltName|title=Title]]", `<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="AltName" title="Title"/></a></p>
 | 
							`<p><a href="`+url+`" rel="nofollow">Name</a></p>`,
 | 
				
			||||||
`)
 | 
							`<p><a href="`+urlWiki+`" rel="nofollow">Name</a></p>`)
 | 
				
			||||||
	test("[[Name|Link.jpg|alt=\"AltName\"|title='Title']]", `<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="AltName" title="Title"/></a></p>
 | 
						test(
 | 
				
			||||||
`)
 | 
							"[[Name|Link.jpg]]",
 | 
				
			||||||
 | 
							`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="Name" title="Name"/></a></p>`,
 | 
				
			||||||
 | 
							`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" alt="Name" title="Name"/></a></p>`)
 | 
				
			||||||
 | 
						test(
 | 
				
			||||||
 | 
							"[[Name|Link.jpg|alt=AltName]]",
 | 
				
			||||||
 | 
							`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="AltName" title="AltName"/></a></p>`,
 | 
				
			||||||
 | 
							`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" alt="AltName" title="AltName"/></a></p>`)
 | 
				
			||||||
 | 
						test(
 | 
				
			||||||
 | 
							"[[Name|Link.jpg|title=Title]]",
 | 
				
			||||||
 | 
							`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="Title" title="Title"/></a></p>`,
 | 
				
			||||||
 | 
							`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" alt="Title" title="Title"/></a></p>`)
 | 
				
			||||||
 | 
						test(
 | 
				
			||||||
 | 
							"[[Name|Link.jpg|alt=AltName|title=Title]]",
 | 
				
			||||||
 | 
							`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="AltName" title="Title"/></a></p>`,
 | 
				
			||||||
 | 
							`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" alt="AltName" title="Title"/></a></p>`)
 | 
				
			||||||
 | 
						test(
 | 
				
			||||||
 | 
							"[[Name|Link.jpg|alt=\"AltName\"|title='Title']]",
 | 
				
			||||||
 | 
							`<p><a href="`+imgurl+`" rel="nofollow"><img src="`+imgurl+`" alt="AltName" title="Title"/></a></p>`,
 | 
				
			||||||
 | 
							`<p><a href="`+imgurlWiki+`" rel="nofollow"><img src="`+imgurlWiki+`" alt="AltName" title="Title"/></a></p>`)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestRender_Commits(t *testing.T) {
 | 
					func TestRender_Commits(t *testing.T) {
 | 
				
			||||||
@@ -236,7 +280,7 @@ func TestRender_Commits(t *testing.T) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	test := func(input, expected string) {
 | 
						test := func(input, expected string) {
 | 
				
			||||||
		buffer := RenderString(input, setting.AppSubURL, nil)
 | 
							buffer := RenderString(input, setting.AppSubURL, nil)
 | 
				
			||||||
		assert.Equal(t, expected, string(buffer))
 | 
							assert.Equal(t, strings.TrimSpace(expected), strings.TrimSpace(string(buffer)))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var sha = "b6dd6210eaebc915fd5be5579c58cce4da2e2579"
 | 
						var sha = "b6dd6210eaebc915fd5be5579c58cce4da2e2579"
 | 
				
			||||||
@@ -245,12 +289,45 @@ func TestRender_Commits(t *testing.T) {
 | 
				
			|||||||
	var tree = strings.Replace(subtree, "/commit/", "/tree/", -1)
 | 
						var tree = strings.Replace(subtree, "/commit/", "/tree/", -1)
 | 
				
			||||||
	var src = strings.Replace(subtree, "/commit/", "/src/", -1)
 | 
						var src = strings.Replace(subtree, "/commit/", "/src/", -1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	test(sha, `<p><a href="`+commit+`" rel="nofollow">b6dd6210ea</a></p>
 | 
						test(sha, `<p><a href="`+commit+`" rel="nofollow">b6dd6210ea</a></p>`)
 | 
				
			||||||
`)
 | 
						test(commit, `<p><a href="`+commit+`" rel="nofollow">b6dd6210ea</a></p>`)
 | 
				
			||||||
	test(commit, `<p><a href="`+commit[5:]+`" rel="nofollow">b6dd6210ea</a></p>
 | 
						test(tree, `<p><a href="`+src+`" rel="nofollow">b6dd6210ea/src</a></p>`)
 | 
				
			||||||
`)
 | 
					}
 | 
				
			||||||
	test(tree, `<p><a href="`+src[5:]+`" rel="nofollow">b6dd6210ea/src</a></p>
 | 
					
 | 
				
			||||||
`)
 | 
					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(string(buffer)))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						url := "../../.images/src/02/train.jpg"
 | 
				
			||||||
 | 
						title := "Train"
 | 
				
			||||||
 | 
						result := 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+`" alt="`+title+`" title="`+title+`"/></a></p>`)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestRender_CrossReferences(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(string(buffer)))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						test(
 | 
				
			||||||
 | 
							"gogits/gogs#12345",
 | 
				
			||||||
 | 
							`<p><a href="`+URLJoin(AppURL, "gogits", "gogs", "issues", "12345")+`" rel="nofollow">gogits/gogs#12345</a></p>`)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestRegExp_MentionPattern(t *testing.T) {
 | 
					func TestRegExp_MentionPattern(t *testing.T) {
 | 
				
			||||||
@@ -387,6 +464,7 @@ func TestRegExp_ShortLinkPattern(t *testing.T) {
 | 
				
			|||||||
func TestRegExp_AnySHA1Pattern(t *testing.T) {
 | 
					func TestRegExp_AnySHA1Pattern(t *testing.T) {
 | 
				
			||||||
	testCases := map[string][]string{
 | 
						testCases := map[string][]string{
 | 
				
			||||||
		"https://github.com/jquery/jquery/blob/a644101ed04d0beacea864ce805e0c4f86ba1cd1/test/unit/event.js#L2703": []string{
 | 
							"https://github.com/jquery/jquery/blob/a644101ed04d0beacea864ce805e0c4f86ba1cd1/test/unit/event.js#L2703": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"github.com",
 | 
								"github.com",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
@@ -396,6 +474,7 @@ func TestRegExp_AnySHA1Pattern(t *testing.T) {
 | 
				
			|||||||
			"L2703",
 | 
								"L2703",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://github.com/jquery/jquery/blob/a644101ed04d0beacea864ce805e0c4f86ba1cd1/test/unit/event.js": []string{
 | 
							"https://github.com/jquery/jquery/blob/a644101ed04d0beacea864ce805e0c4f86ba1cd1/test/unit/event.js": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"github.com",
 | 
								"github.com",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
@@ -405,6 +484,7 @@ func TestRegExp_AnySHA1Pattern(t *testing.T) {
 | 
				
			|||||||
			"",
 | 
								"",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://github.com/jquery/jquery/commit/0705be475092aede1eddae01319ec931fb9c65fc": []string{
 | 
							"https://github.com/jquery/jquery/commit/0705be475092aede1eddae01319ec931fb9c65fc": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"github.com",
 | 
								"github.com",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
@@ -414,6 +494,7 @@ func TestRegExp_AnySHA1Pattern(t *testing.T) {
 | 
				
			|||||||
			"",
 | 
								"",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://github.com/jquery/jquery/tree/0705be475092aede1eddae01319ec931fb9c65fc/src": []string{
 | 
							"https://github.com/jquery/jquery/tree/0705be475092aede1eddae01319ec931fb9c65fc/src": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"github.com",
 | 
								"github.com",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
			"jquery",
 | 
								"jquery",
 | 
				
			||||||
@@ -423,6 +504,7 @@ func TestRegExp_AnySHA1Pattern(t *testing.T) {
 | 
				
			|||||||
			"",
 | 
								"",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://try.gogs.io/gogs/gogs/commit/d8a994ef243349f321568f9e36d5c3f444b99cae#diff-2": []string{
 | 
							"https://try.gogs.io/gogs/gogs/commit/d8a994ef243349f321568f9e36d5c3f444b99cae#diff-2": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"try.gogs.io",
 | 
								"try.gogs.io",
 | 
				
			||||||
			"gogs",
 | 
								"gogs",
 | 
				
			||||||
			"gogs",
 | 
								"gogs",
 | 
				
			||||||
@@ -441,30 +523,35 @@ func TestRegExp_AnySHA1Pattern(t *testing.T) {
 | 
				
			|||||||
func TestRegExp_IssueFullPattern(t *testing.T) {
 | 
					func TestRegExp_IssueFullPattern(t *testing.T) {
 | 
				
			||||||
	testCases := map[string][]string{
 | 
						testCases := map[string][]string{
 | 
				
			||||||
		"https://github.com/gogits/gogs/pull/3244": []string{
 | 
							"https://github.com/gogits/gogs/pull/3244": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"github.com/gogits/gogs/pull/",
 | 
								"github.com/gogits/gogs/pull/",
 | 
				
			||||||
			"3244",
 | 
								"3244",
 | 
				
			||||||
			"",
 | 
								"",
 | 
				
			||||||
			"",
 | 
								"",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://github.com/gogits/gogs/issues/3247#issuecomment-231517079": []string{
 | 
							"https://github.com/gogits/gogs/issues/3247#issuecomment-231517079": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"github.com/gogits/gogs/issues/",
 | 
								"github.com/gogits/gogs/issues/",
 | 
				
			||||||
			"3247",
 | 
								"3247",
 | 
				
			||||||
			"#issuecomment-231517079",
 | 
								"#issuecomment-231517079",
 | 
				
			||||||
			"",
 | 
								"",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://try.gogs.io/gogs/gogs/issues/4#issue-685": []string{
 | 
							"https://try.gogs.io/gogs/gogs/issues/4#issue-685": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"try.gogs.io/gogs/gogs/issues/",
 | 
								"try.gogs.io/gogs/gogs/issues/",
 | 
				
			||||||
			"4",
 | 
								"4",
 | 
				
			||||||
			"#issue-685",
 | 
								"#issue-685",
 | 
				
			||||||
			"",
 | 
								"",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://youtrack.jetbrains.com/issue/JT-36485": []string{
 | 
							"https://youtrack.jetbrains.com/issue/JT-36485": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"youtrack.jetbrains.com/issue/",
 | 
								"youtrack.jetbrains.com/issue/",
 | 
				
			||||||
			"JT-36485",
 | 
								"JT-36485",
 | 
				
			||||||
			"",
 | 
								"",
 | 
				
			||||||
			"",
 | 
								"",
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		"https://youtrack.jetbrains.com/issue/JT-36485#comment=27-1508676": []string{
 | 
							"https://youtrack.jetbrains.com/issue/JT-36485#comment=27-1508676": []string{
 | 
				
			||||||
 | 
								"https",
 | 
				
			||||||
			"youtrack.jetbrains.com/issue/",
 | 
								"youtrack.jetbrains.com/issue/",
 | 
				
			||||||
			"JT-36485",
 | 
								"JT-36485",
 | 
				
			||||||
			"#comment=27-1508676",
 | 
								"#comment=27-1508676",
 | 
				
			||||||
@@ -549,23 +636,6 @@ Ideas and codes
 | 
				
			|||||||
- Node graph editors https://github.com/ocornut/imgui/issues/306
 | 
					- Node graph editors https://github.com/ocornut/imgui/issues/306
 | 
				
			||||||
- [[Memory Editor|memory_editor_example]]
 | 
					- [[Memory Editor|memory_editor_example]]
 | 
				
			||||||
- [[Plot var helper|plot_var_example]]`,
 | 
					- [[Plot var helper|plot_var_example]]`,
 | 
				
			||||||
	// rendered
 | 
					 | 
				
			||||||
	`<p>Wiki! Enjoy :)</p>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
<ul>
 | 
					 | 
				
			||||||
<li><a href="` + AppSubURL + `wiki/Links" rel="nofollow">Links, Language bindings, Engine bindings</a></li>
 | 
					 | 
				
			||||||
<li><a href="` + AppSubURL + `wiki/Tips" rel="nofollow">Tips</a></li>
 | 
					 | 
				
			||||||
</ul>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
<p>Ideas and codes</p>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
<ul>
 | 
					 | 
				
			||||||
<li>Bezier widget (by <a href="` + AppURL + `r-lyeh" rel="nofollow">@r-lyeh</a>)<a href="` + AppSubURL + `issues/786" rel="nofollow">#786</a></li>
 | 
					 | 
				
			||||||
<li>Node graph editors<a href="` + AppSubURL + `issues/306" rel="nofollow">#306</a></li>
 | 
					 | 
				
			||||||
<li><a href="` + AppSubURL + `wiki/memory_editor_example" rel="nofollow">Memory Editor</a></li>
 | 
					 | 
				
			||||||
<li><a href="` + AppSubURL + `wiki/plot_var_example" rel="nofollow">Plot var helper</a></li>
 | 
					 | 
				
			||||||
</ul>
 | 
					 | 
				
			||||||
`,
 | 
					 | 
				
			||||||
	// wine-staging wiki home extract: tables, special wiki syntax, images
 | 
						// wine-staging wiki home extract: tables, special wiki syntax, images
 | 
				
			||||||
	`## What is Wine Staging?
 | 
						`## What is Wine Staging?
 | 
				
			||||||
**Wine Staging** on website [wine-staging.com](http://wine-staging.com).
 | 
					**Wine Staging** on website [wine-staging.com](http://wine-staging.com).
 | 
				
			||||||
@@ -576,11 +646,35 @@ Here are some links to the most important topics. You can find the full list of
 | 
				
			|||||||
| [[images/icon-install.png]]    | [[Installation]]                                         |
 | 
					| [[images/icon-install.png]]    | [[Installation]]                                         |
 | 
				
			||||||
|--------------------------------|----------------------------------------------------------|
 | 
					|--------------------------------|----------------------------------------------------------|
 | 
				
			||||||
| [[images/icon-usage.png]]      | [[Usage]]                                                |
 | 
					| [[images/icon-usage.png]]      | [[Usage]]                                                |
 | 
				
			||||||
| [[images/icon-config.png]]     | [[Configuration]]                                        |
 | 
					 | 
				
			||||||
| [[images/icon-bug.png]]        | [Bugs](http://bugs.wine-staging.com)                     |
 | 
					 | 
				
			||||||
`,
 | 
					`,
 | 
				
			||||||
	// rendered
 | 
						// libgdx wiki page: inline images with special syntax
 | 
				
			||||||
	`<h2>What is Wine Staging?</h2>
 | 
						`[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]]`,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					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>Ideas and codes</p>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<ul>
 | 
				
			||||||
 | 
					<li>Bezier widget (by <a href="` + AppURL + `r-lyeh" rel="nofollow">@r-lyeh</a>)<a href="https://github.com/ocornut/imgui/issues/786" rel="nofollow">#786</a></li>
 | 
				
			||||||
 | 
					<li>Node graph editors<a href="https://github.com/ocornut/imgui/issues/306" rel="nofollow">#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>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>
 | 
					<p><strong>Wine Staging</strong> on website <a href="http://wine-staging.com" rel="nofollow">wine-staging.com</a>.</p>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -591,66 +685,53 @@ Here are some links to the most important topics. You can find the full list of
 | 
				
			|||||||
<table>
 | 
					<table>
 | 
				
			||||||
<thead>
 | 
					<thead>
 | 
				
			||||||
<tr>
 | 
					<tr>
 | 
				
			||||||
<th><a href="` + AppSubURL + `wiki/raw/images%2Ficon-install.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images%2Ficon-install.png" alt="images/icon-install.png" title="icon-install.png"/></a></th>
 | 
					<th><a href="` + baseURLImages + `images/icon-install.png" rel="nofollow"><img src="` + baseURLImages + `images/icon-install.png" alt="images/icon-install.png" title="icon-install.png"/></a></th>
 | 
				
			||||||
<th><a href="` + AppSubURL + `wiki/Installation" rel="nofollow">Installation</a></th>
 | 
					<th><a href="` + baseURLContent + `Installation" rel="nofollow">Installation</a></th>
 | 
				
			||||||
</tr>
 | 
					</tr>
 | 
				
			||||||
</thead>
 | 
					</thead>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<tbody>
 | 
					<tbody>
 | 
				
			||||||
<tr>
 | 
					<tr>
 | 
				
			||||||
<td><a href="` + AppSubURL + `wiki/raw/images%2Ficon-usage.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images%2Ficon-usage.png" alt="images/icon-usage.png" title="icon-usage.png"/></a></td>
 | 
					<td><a href="` + baseURLImages + `images/icon-usage.png" rel="nofollow"><img src="` + baseURLImages + `images/icon-usage.png" alt="images/icon-usage.png" title="icon-usage.png"/></a></td>
 | 
				
			||||||
<td><a href="` + AppSubURL + `wiki/Usage" rel="nofollow">Usage</a></td>
 | 
					<td><a href="` + baseURLContent + `Usage" rel="nofollow">Usage</a></td>
 | 
				
			||||||
</tr>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
<tr>
 | 
					 | 
				
			||||||
<td><a href="` + AppSubURL + `wiki/raw/images%2Ficon-config.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images%2Ficon-config.png" alt="images/icon-config.png" title="icon-config.png"/></a></td>
 | 
					 | 
				
			||||||
<td><a href="` + AppSubURL + `wiki/Configuration" rel="nofollow">Configuration</a></td>
 | 
					 | 
				
			||||||
</tr>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
<tr>
 | 
					 | 
				
			||||||
<td><a href="` + AppSubURL + `wiki/raw/images%2Ficon-bug.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images%2Ficon-bug.png" alt="images/icon-bug.png" title="icon-bug.png"/></a></td>
 | 
					 | 
				
			||||||
<td><a href="http://bugs.wine-staging.com" rel="nofollow">Bugs</a></td>
 | 
					 | 
				
			||||||
</tr>
 | 
					</tr>
 | 
				
			||||||
</tbody>
 | 
					</tbody>
 | 
				
			||||||
</table>
 | 
					</table>
 | 
				
			||||||
`,
 | 
					`,
 | 
				
			||||||
	// libgdx wiki page: inline images with special syntax
 | 
							`<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>
 | 
				
			||||||
	`[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]]`,
 | 
					 | 
				
			||||||
	// rendered
 | 
					 | 
				
			||||||
	`<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>
 | 
					<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>
 | 
					<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="` + AppSubURL + `wiki/raw/images%2F1.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images%2F1.png" alt="images/1.png" title="1.png"/></a></li>
 | 
					<a href="` + baseURLImages + `images/1.png" rel="nofollow"><img src="` + baseURLImages + `images/1.png" alt="images/1.png" title="1.png"/></a></li>
 | 
				
			||||||
<li>Perform a test run by hitting the Run! button.
 | 
					<li>Perform a test run by hitting the Run! button.
 | 
				
			||||||
<a href="` + AppSubURL + `wiki/raw/images%2F2.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images%2F2.png" alt="images/2.png" title="2.png"/></a></li>
 | 
					<a href="` + baseURLImages + `images/2.png" rel="nofollow"><img src="` + baseURLImages + `images/2.png" alt="images/2.png" title="2.png"/></a></li>
 | 
				
			||||||
</ol>
 | 
					</ol>
 | 
				
			||||||
`,
 | 
					`,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestTotal_RenderString(t *testing.T) {
 | 
					func TestTotal_RenderString(t *testing.T) {
 | 
				
			||||||
	for i := 0; i < len(sameCases); i += 2 {
 | 
						answers := testAnswers(URLJoin(AppSubURL, "src", "master/"), URLJoin(AppSubURL, "raw", "master/"))
 | 
				
			||||||
		line := RenderString(sameCases[i], AppSubURL, map[string]string{})
 | 
					
 | 
				
			||||||
		assert.Equal(t, sameCases[i+1], line)
 | 
						for i := 0; i < len(sameCases); i++ {
 | 
				
			||||||
 | 
							line := RenderString(sameCases[i], URLJoin(AppSubURL, "src", "master/"), nil)
 | 
				
			||||||
 | 
							assert.Equal(t, answers[i], line)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	testCases := []string{}
 | 
						testCases := []string{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for i := 0; i < len(testCases); i += 2 {
 | 
						for i := 0; i < len(testCases); i += 2 {
 | 
				
			||||||
		line := RenderString(testCases[i], AppSubURL, map[string]string{})
 | 
							line := RenderString(testCases[i], AppSubURL, nil)
 | 
				
			||||||
		assert.Equal(t, testCases[i+1], line)
 | 
							assert.Equal(t, testCases[i+1], line)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestTotal_RenderWiki(t *testing.T) {
 | 
					func TestTotal_RenderWiki(t *testing.T) {
 | 
				
			||||||
	for i := 0; i < len(sameCases); i += 2 {
 | 
						answers := testAnswers(URLJoin(AppSubURL, "wiki/"), URLJoin(AppSubURL, "wiki", "raw/"))
 | 
				
			||||||
		line := RenderWiki([]byte(sameCases[i]), AppSubURL, map[string]string{})
 | 
					
 | 
				
			||||||
		assert.Equal(t, sameCases[i+1], line)
 | 
						for i := 0; i < len(sameCases); i++ {
 | 
				
			||||||
 | 
							line := RenderWiki([]byte(sameCases[i]), AppSubURL, nil)
 | 
				
			||||||
 | 
							assert.Equal(t, answers[i], line)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	testCases := []string{
 | 
						testCases := []string{
 | 
				
			||||||
@@ -667,7 +748,7 @@ func TestTotal_RenderWiki(t *testing.T) {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for i := 0; i < len(testCases); i += 2 {
 | 
						for i := 0; i < len(testCases); i += 2 {
 | 
				
			||||||
		line := RenderWiki([]byte(testCases[i]), AppSubURL, map[string]string{})
 | 
							line := RenderWiki([]byte(testCases[i]), AppSubURL, nil)
 | 
				
			||||||
		assert.Equal(t, testCases[i+1], line)
 | 
							assert.Equal(t, testCases[i+1], line)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -27,7 +27,13 @@ func Markdown(ctx *context.APIContext, form api.MarkdownOption) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	switch form.Mode {
 | 
						switch form.Mode {
 | 
				
			||||||
	case "gfm":
 | 
						case "gfm":
 | 
				
			||||||
		ctx.Write(markdown.Render([]byte(form.Text), markdown.URLJoin(setting.AppURL, form.Context), nil))
 | 
							md := []byte(form.Text)
 | 
				
			||||||
 | 
							context := markdown.URLJoin(setting.AppURL, form.Context)
 | 
				
			||||||
 | 
							if form.Wiki {
 | 
				
			||||||
 | 
								ctx.Write([]byte(markdown.RenderWiki(md, context, nil)))
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								ctx.Write(markdown.Render(md, context, nil))
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	default:
 | 
						default:
 | 
				
			||||||
		ctx.Write(markdown.RenderRaw([]byte(form.Text), "", false))
 | 
							ctx.Write(markdown.RenderRaw([]byte(form.Text), "", false))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -53,6 +53,7 @@ func TestAPI_RenderGFM(t *testing.T) {
 | 
				
			|||||||
		Mode:    "gfm",
 | 
							Mode:    "gfm",
 | 
				
			||||||
		Text:    "",
 | 
							Text:    "",
 | 
				
			||||||
		Context: Repo,
 | 
							Context: Repo,
 | 
				
			||||||
 | 
							Wiki:    true,
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	requrl, _ := url.Parse(markdown.URLJoin(AppURL, "api", "v1", "markdown"))
 | 
						requrl, _ := url.Parse(markdown.URLJoin(AppURL, "api", "v1", "markdown"))
 | 
				
			||||||
	req := &http.Request{
 | 
						req := &http.Request{
 | 
				
			||||||
@@ -74,7 +75,7 @@ func TestAPI_RenderGFM(t *testing.T) {
 | 
				
			|||||||
<ul>
 | 
					<ul>
 | 
				
			||||||
<li><a href="` + AppSubURL + `wiki/Links" rel="nofollow">Links, Language bindings, Engine bindings</a></li>
 | 
					<li><a href="` + AppSubURL + `wiki/Links" rel="nofollow">Links, Language bindings, Engine bindings</a></li>
 | 
				
			||||||
<li><a href="` + AppSubURL + `wiki/Tips" rel="nofollow">Tips</a></li>
 | 
					<li><a href="` + AppSubURL + `wiki/Tips" rel="nofollow">Tips</a></li>
 | 
				
			||||||
<li>Bezier widget (by <a href="` + AppURL + `r-lyeh" rel="nofollow">@r-lyeh</a>)<a href="` + AppSubURL + `issues/786" rel="nofollow">#786</a></li>
 | 
					<li>Bezier widget (by <a href="` + AppURL + `r-lyeh" rel="nofollow">@r-lyeh</a>)<a href="https://github.com/ocornut/imgui/issues/786" rel="nofollow">#786</a></li>
 | 
				
			||||||
</ul>
 | 
					</ul>
 | 
				
			||||||
`,
 | 
					`,
 | 
				
			||||||
		// wine-staging wiki home extract: special wiki syntax, images
 | 
							// wine-staging wiki home extract: special wiki syntax, images
 | 
				
			||||||
@@ -97,7 +98,7 @@ Here are some links to the most important topics. You can find the full list of
 | 
				
			|||||||
<p>Here are some links to the most important topics. You can find the full list of pages at the sidebar.</p>
 | 
					<p>Here are some links to the most important topics. You can find the full list of pages at the sidebar.</p>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<p><a href="` + AppSubURL + `wiki/Configuration" rel="nofollow">Configuration</a>
 | 
					<p><a href="` + AppSubURL + `wiki/Configuration" rel="nofollow">Configuration</a>
 | 
				
			||||||
<a href="` + AppSubURL + `wiki/raw/images%2Ficon-bug.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images%2Ficon-bug.png" alt="images/icon-bug.png" title="icon-bug.png"/></a></p>
 | 
					<a href="` + AppSubURL + `wiki/raw/images/icon-bug.png" rel="nofollow"><img src="` + AppSubURL + `wiki/raw/images/icon-bug.png" alt="images/icon-bug.png" title="icon-bug.png"/></a></p>
 | 
				
			||||||
`,
 | 
					`,
 | 
				
			||||||
		// Guard wiki sidebar: special syntax
 | 
							// Guard wiki sidebar: special syntax
 | 
				
			||||||
		`[[Guardfile-DSL / Configuring-Guard|Guardfile-DSL---Configuring-Guard]]`,
 | 
							`[[Guardfile-DSL / Configuring-Guard|Guardfile-DSL---Configuring-Guard]]`,
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										1
									
								
								vendor/code.gitea.io/sdk/gitea/miscellaneous.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								vendor/code.gitea.io/sdk/gitea/miscellaneous.go
									
									
									
										generated
									
									
										vendored
									
									
								
							@@ -9,4 +9,5 @@ type MarkdownOption struct {
 | 
				
			|||||||
	Text    string
 | 
						Text    string
 | 
				
			||||||
	Mode    string
 | 
						Mode    string
 | 
				
			||||||
	Context string
 | 
						Context string
 | 
				
			||||||
 | 
						Wiki    bool
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user