diff --git a/modules/markup/html.go b/modules/markup/html.go index b436ff6c3d309..1eedf095a0c32 100644 --- a/modules/markup/html.go +++ b/modules/markup/html.go @@ -761,10 +761,10 @@ func shortLinkProcessor(ctx *RenderContext, node *html.Node) { if image { link = strings.ReplaceAll(link, " ", "+") } else { - link = strings.ReplaceAll(link, " ", "-") + link = strings.ReplaceAll(link, " ", "-") // FIXME: it should support dashes in the link, eg: "the-dash-support.-" } if !strings.Contains(link, "/") { - link = url.PathEscape(link) + link = url.PathEscape(link) // FIXME: it doesn't seem right and it might cause double-escaping } } if image { @@ -796,28 +796,7 @@ func shortLinkProcessor(ctx *RenderContext, node *html.Node) { childNode.Attr = childNode.Attr[:2] } } else { - if !absoluteLink { - var base string - if ctx.IsWiki { - switch ext { - case "": - // no file extension, create a regular wiki link - base = ctx.Links.WikiLink() - default: - // we have a file extension: - // return a regular wiki link if it's a renderable file (extension), - // raw link otherwise - if Type(link) != "" { - base = ctx.Links.WikiLink() - } else { - base = ctx.Links.WikiRawLink() - } - } - } else { - base = ctx.Links.SrcLink() - } - link = util.URLJoin(base, link) - } + link, _ = ResolveLink(ctx, link, "") childNode.Type = html.TextNode childNode.Data = name } diff --git a/modules/markup/html_link.go b/modules/markup/html_link.go new file mode 100644 index 0000000000000..a41b87e9fa83d --- /dev/null +++ b/modules/markup/html_link.go @@ -0,0 +1,35 @@ +// Copyright 2024 The Gitea Authors. All rights reserved. +// SPDX-License-Identifier: MIT + +package markup + +import ( + "path" + + "code.gitea.io/gitea/modules/util" +) + +func ResolveLink(ctx *RenderContext, link, userContentAnchorPrefix string) (result string, resolved bool) { + isAnchorFragment := link != "" && link[0] == '#' + if !isAnchorFragment && !IsFullURLString(link) { + linkBase := ctx.Links.Base + if ctx.IsWiki { + if ext := path.Ext(link); ext == "" || ext == ".-" { + linkBase = ctx.Links.WikiLink() // the link is for a wiki page + } else if DetectMarkupTypeByFileName(link) != "" { + linkBase = ctx.Links.WikiLink() // the link is renderable as a wiki page + } else { + linkBase = ctx.Links.WikiRawLink() // otherwise, use a raw link instead to view&download medias + } + } else if ctx.Links.BranchPath != "" || ctx.Links.TreePath != "" { + // if there is no BranchPath, then the link will be something like "/owner/repo/src/{the-file-path}" + // and then this link will be handled by the "legacy-ref" code and be redirected to the default branch like "/owner/repo/src/branch/main/{the-file-path}" + linkBase = ctx.Links.SrcLink() + } + link, resolved = util.URLJoin(linkBase, link), true + } + if isAnchorFragment && userContentAnchorPrefix != "" { + link, resolved = userContentAnchorPrefix+link[1:], true + } + return link, resolved +} diff --git a/modules/markup/html_test.go b/modules/markup/html_test.go index a0642bcfa4127..64cc30d246883 100644 --- a/modules/markup/html_test.go +++ b/modules/markup/html_test.go @@ -442,6 +442,10 @@ func TestRender_ShortLinks(t *testing.T) { "[[Link]]", `

Link

`, `

Link

`) + test( + "[[Link.-]]", + `

Link.-

`, + `

Link.-

`) test( "[[Link.jpg]]", `

Link.jpg

`, diff --git a/modules/markup/markdown/goldmark.go b/modules/markup/markdown/goldmark.go index b8b3aeaab0ec5..ab11a56c4d9f1 100644 --- a/modules/markup/markdown/goldmark.go +++ b/modules/markup/markdown/goldmark.go @@ -67,7 +67,7 @@ func (g *ASTTransformer) Transform(node *ast.Document, reader text.Reader, pc pa case *ast.Image: g.transformImage(ctx, v, reader) case *ast.Link: - g.transformLink(ctx, v, reader) + g.transformLink(ctx, v) case *ast.List: g.transformList(ctx, v, reader, rc) case *ast.Text: diff --git a/modules/markup/markdown/markdown_test.go b/modules/markup/markdown/markdown_test.go index cefbdbf9c2b8e..ecd747214b12e 100644 --- a/modules/markup/markdown/markdown_test.go +++ b/modules/markup/markdown/markdown_test.go @@ -626,7 +626,7 @@ mail@domain.com https://example.com/file.bin
local link
remote link
-local link
+local link
remote link
local image
local image
@@ -682,7 +682,7 @@ space

https://example.com/file.bin
local link
remote link
-local link
+local link
remote link
local image
local image
@@ -740,7 +740,7 @@ space

https://example.com/file.bin
local link
remote link
-local link
+local link
remote link
local image
local image
@@ -857,7 +857,7 @@ space

Expected: `

space @mention-user
/just/a/path.bin
https://example.com/file.bin
-local link
+local link
remote link
local link
remote link
@@ -975,7 +975,7 @@ space

for i, c := range cases { result, err := markdown.RenderString(&markup.RenderContext{Ctx: context.Background(), Links: c.Links, IsWiki: c.IsWiki}, input) assert.NoError(t, err, "Unexpected error in testcase: %v", i) - assert.Equal(t, template.HTML(c.Expected), result, "Unexpected result in testcase %v", i) + assert.Equal(t, c.Expected, string(result), "Unexpected result in testcase %v", i) } } diff --git a/modules/markup/markdown/transform_link.go b/modules/markup/markdown/transform_link.go index 7e305b74bcdb1..38fbf693ab837 100644 --- a/modules/markup/markdown/transform_link.go +++ b/modules/markup/markdown/transform_link.go @@ -4,39 +4,13 @@ package markdown import ( - "path/filepath" - "code.gitea.io/gitea/modules/markup" - giteautil "code.gitea.io/gitea/modules/util" "github.com/yuin/goldmark/ast" - "github.com/yuin/goldmark/text" ) -func (g *ASTTransformer) transformLink(ctx *markup.RenderContext, v *ast.Link, reader text.Reader) { - // Links need their href to munged to be a real value - link := v.Destination - isAnchorFragment := len(link) > 0 && link[0] == '#' - if !isAnchorFragment && !markup.IsFullURLBytes(link) { - base := ctx.Links.Base - if ctx.IsWiki { - if filepath.Ext(string(link)) == "" { - // This link doesn't have a file extension - assume a regular wiki link - base = ctx.Links.WikiLink() - } else if markup.Type(string(link)) != "" { - // If it's a file type we can render, use a regular wiki link - base = ctx.Links.WikiLink() - } else { - // Otherwise, use a raw link instead - base = ctx.Links.WikiRawLink() - } - } else if ctx.Links.HasBranchInfo() { - base = ctx.Links.SrcLink() - } - link = []byte(giteautil.URLJoin(base, string(link))) - } - if isAnchorFragment { - link = []byte("#user-content-" + string(link)[1:]) +func (g *ASTTransformer) transformLink(ctx *markup.RenderContext, v *ast.Link) { + if link, resolved := markup.ResolveLink(ctx, string(v.Destination), "#user-content-"); resolved { + v.Destination = []byte(link) } - v.Destination = link } diff --git a/modules/markup/renderer.go b/modules/markup/renderer.go index 3284a8194e7a6..c18ba918daa02 100644 --- a/modules/markup/renderer.go +++ b/modules/markup/renderer.go @@ -370,22 +370,14 @@ func renderFile(ctx *RenderContext, input io.Reader, output io.Writer) error { return ErrUnsupportedRenderExtension{extension} } -// Type returns if markup format via the filename -func Type(filename string) string { +// DetectMarkupTypeByFileName returns the possible markup format type via the filename +func DetectMarkupTypeByFileName(filename string) string { if parser := GetRendererByFileName(filename); parser != nil { return parser.Name() } return "" } -// IsMarkupFile reports whether file is a markup type file -func IsMarkupFile(name, markup string) bool { - if parser := GetRendererByFileName(name); parser != nil { - return parser.Name() == markup - } - return false -} - func PreviewableExtensions() []string { extensions := make([]string, 0, len(extRenderers)) for extension := range extRenderers { diff --git a/modules/templates/util_render_test.go b/modules/templates/util_render_test.go index f493b899e393c..32e53b5215574 100644 --- a/modules/templates/util_render_test.go +++ b/modules/templates/util_render_test.go @@ -174,7 +174,7 @@ func TestRenderMarkdownToHtml(t *testing.T) { https://example.com/file.bin local link remote link -local link +local link remote link local image remote image @@ -190,7 +190,7 @@ com 88fc37a3c0a4dda553bdcfc80c178a58247f42fb mit #123 space

` - assert.EqualValues(t, expected, RenderMarkdownToHtml(context.Background(), testInput())) + assert.Equal(t, expected, string(RenderMarkdownToHtml(context.Background(), testInput()))) } func TestRenderLabels(t *testing.T) { diff --git a/routers/web/repo/render.go b/routers/web/repo/render.go index e64db03e2015e..6aba9e0ac1436 100644 --- a/routers/web/repo/render.go +++ b/routers/web/repo/render.go @@ -47,7 +47,7 @@ func RenderFile(ctx *context.Context) { rd := charset.ToUTF8WithFallbackReader(io.MultiReader(bytes.NewReader(buf), dataRc), charset.ConvertOpts{}) ctx.Resp.Header().Add("Content-Security-Policy", "frame-src 'self'; sandbox allow-scripts") - if markupType := markup.Type(blob.Name()); markupType == "" { + if markupType := markup.DetectMarkupTypeByFileName(blob.Name()); markupType == "" { if isTextFile { _, _ = io.Copy(ctx.Resp, rd) } else { diff --git a/routers/web/repo/view.go b/routers/web/repo/view.go index 386ef7be5ce87..da849d7b72731 100644 --- a/routers/web/repo/view.go +++ b/routers/web/repo/view.go @@ -307,7 +307,7 @@ func renderReadmeFile(ctx *context.Context, subfolder string, readmeFile *git.Tr rd := charset.ToUTF8WithFallbackReader(io.MultiReader(bytes.NewReader(buf), dataRc), charset.ConvertOpts{}) - if markupType := markup.Type(readmeFile.Name()); markupType != "" { + if markupType := markup.DetectMarkupTypeByFileName(readmeFile.Name()); markupType != "" { ctx.Data["IsMarkup"] = true ctx.Data["MarkupType"] = markupType @@ -499,7 +499,7 @@ func renderFile(ctx *context.Context, entry *git.TreeEntry) { readmeExist := util.IsReadmeFileName(blob.Name()) ctx.Data["ReadmeExist"] = readmeExist - markupType := markup.Type(blob.Name()) + markupType := markup.DetectMarkupTypeByFileName(blob.Name()) // If the markup is detected by custom markup renderer it should not be reset later on // to not pass it down to the render context. detected := false @@ -606,9 +606,9 @@ func renderFile(ctx *context.Context, entry *git.TreeEntry) { break } - // TODO: this logic seems strange, it duplicates with "isRepresentableAsText=true", it is not the same as "LFSFileGet" in "lfs.go" - // maybe for this case, the file is a binary file, and shouldn't be rendered? - if markupType := markup.Type(blob.Name()); markupType != "" { + // TODO: this logic duplicates with "isRepresentableAsText=true", it is not the same as "LFSFileGet" in "lfs.go" + // It is used by "external renders", markupRender will execute external programs to get rendered content. + if markupType := markup.DetectMarkupTypeByFileName(blob.Name()); markupType != "" { rd := io.MultiReader(bytes.NewReader(buf), dataRc) ctx.Data["IsMarkup"] = true ctx.Data["MarkupType"] = markupType diff --git a/routers/web/repo/wiki.go b/routers/web/repo/wiki.go index df15f61b173ff..ff6397cd2a82e 100644 --- a/routers/web/repo/wiki.go +++ b/routers/web/repo/wiki.go @@ -532,7 +532,7 @@ func Wiki(ctx *context.Context) { } wikiPath := entry.Name() - if markup.Type(wikiPath) != markdown.MarkupName { + if markup.DetectMarkupTypeByFileName(wikiPath) != markdown.MarkupName { ext := strings.ToUpper(filepath.Ext(wikiPath)) ctx.Data["FormatWarning"] = fmt.Sprintf("%s rendering is not supported at the moment. Rendered as Markdown.", ext) }