mattermost-community-enterp.../public/shared/markdown/html.go
Claude ec1f89217a Merge: Complete Mattermost Server with Community Enterprise
Full Mattermost server source with integrated Community Enterprise features.
Includes vendor directory for offline/air-gapped builds.

Structure:
- enterprise-impl/: Enterprise feature implementations
- enterprise-community/: Init files that register implementations
- enterprise/: Bridge imports (community_imports.go)
- vendor/: All dependencies for offline builds

Build (online):
  go build ./cmd/mattermost

Build (offline/air-gapped):
  go build -mod=vendor ./cmd/mattermost

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-17 23:59:07 +09:00

214 lines
6.2 KiB
Go

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
package markdown
import (
"fmt"
"strings"
)
var htmlEscaper = strings.NewReplacer(
`&`, "&amp;",
`<`, "&lt;",
`>`, "&gt;",
`"`, "&quot;",
)
// RenderHTML produces HTML with the same behavior as the example renderer used in the CommonMark
// reference materials except for one slight difference: for brevity, no unnecessary whitespace is
// inserted between elements. The output is not defined by the CommonMark spec, and it exists
// primarily as an aid in testing.
func RenderHTML(markdown string) string {
return RenderBlockHTML(Parse(markdown))
}
func RenderBlockHTML(block Block, referenceDefinitions []*ReferenceDefinition) (result string) {
return renderBlockHTML(block, referenceDefinitions, false)
}
func renderBlockHTML(block Block, referenceDefinitions []*ReferenceDefinition, isTightList bool) string {
var resultSb strings.Builder
switch v := block.(type) {
case *Document:
for _, block := range v.Children {
resultSb.WriteString(RenderBlockHTML(block, referenceDefinitions))
}
case *Paragraph:
if len(v.Text) == 0 {
return ""
}
if !isTightList {
resultSb.WriteString("<p>")
}
for _, inline := range v.ParseInlines(referenceDefinitions) {
resultSb.WriteString(RenderInlineHTML(inline))
}
if !isTightList {
resultSb.WriteString("</p>")
}
case *List:
if v.IsOrdered {
if v.OrderedStart != 1 {
resultSb.WriteString(fmt.Sprintf(`<ol start="%v">`, v.OrderedStart))
} else {
resultSb.WriteString("<ol>")
}
} else {
resultSb.WriteString("<ul>")
}
for _, block := range v.Children {
resultSb.WriteString(renderBlockHTML(block, referenceDefinitions, !v.IsLoose))
}
if v.IsOrdered {
resultSb.WriteString("</ol>")
} else {
resultSb.WriteString("</ul>")
}
case *ListItem:
resultSb.WriteString("<li>")
for _, block := range v.Children {
resultSb.WriteString(renderBlockHTML(block, referenceDefinitions, isTightList))
}
resultSb.WriteString("</li>")
case *BlockQuote:
resultSb.WriteString("<blockquote>")
for _, block := range v.Children {
resultSb.WriteString(RenderBlockHTML(block, referenceDefinitions))
}
resultSb.WriteString("</blockquote>")
case *FencedCode:
if info := v.Info(); info != "" {
language := strings.Fields(info)[0]
resultSb.WriteString(`<pre><code class="language-` + htmlEscaper.Replace(language) + `">`)
} else {
resultSb.WriteString("<pre><code>")
}
resultSb.WriteString(htmlEscaper.Replace(v.Code()))
resultSb.WriteString("</code></pre>")
case *IndentedCode:
resultSb.WriteString("<pre><code>")
resultSb.WriteString(htmlEscaper.Replace(v.Code()))
resultSb.WriteString("</code></pre>")
default:
panic(fmt.Sprintf("missing case for type %T", v))
}
return resultSb.String()
}
func escapeURL(url string) (result string) {
for i := 0; i < len(url); {
switch b := url[i]; b {
case ';', '/', '?', ':', '@', '&', '=', '+', '$', ',', '-', '_', '.', '!', '~', '*', '\'', '(', ')', '#':
result += string(b)
i++
default:
if b == '%' && i+2 < len(url) && isHexByte(url[i+1]) && isHexByte(url[i+2]) {
result += url[i : i+3]
i += 3
} else if (b >= 'a' && b <= 'z') || (b >= 'A' && b <= 'Z') || (b >= '0' && b <= '9') {
result += string(b)
i++
} else {
result += fmt.Sprintf("%%%0X", b)
i++
}
}
}
return
}
func RenderInlineHTML(inline Inline) (result string) {
switch v := inline.(type) {
case *Text:
return htmlEscaper.Replace(v.Text)
case *HardLineBreak:
return "<br />"
case *SoftLineBreak:
return "\n"
case *CodeSpan:
return "<code>" + htmlEscaper.Replace(v.Code) + "</code>"
case *InlineImage:
result += `<img src="` + htmlEscaper.Replace(escapeURL(v.Destination())) + `" alt="` + htmlEscaper.Replace(renderImageAltText(v.Children)) + `"`
if title := v.Title(); title != "" {
result += ` title="` + htmlEscaper.Replace(title) + `"`
}
result += ` />`
case *ReferenceImage:
result += `<img src="` + htmlEscaper.Replace(escapeURL(v.Destination())) + `" alt="` + htmlEscaper.Replace(renderImageAltText(v.Children)) + `"`
if title := v.Title(); title != "" {
result += ` title="` + htmlEscaper.Replace(title) + `"`
}
result += ` />`
case *InlineLink:
var resultSb strings.Builder
resultSb.WriteString(`<a href="` + htmlEscaper.Replace(escapeURL(v.Destination())) + `"`)
if title := v.Title(); title != "" {
resultSb.WriteString(` title="` + htmlEscaper.Replace(title) + `"`)
}
resultSb.WriteString(`>`)
for _, inline := range v.Children {
resultSb.WriteString(RenderInlineHTML(inline))
}
resultSb.WriteString("</a>")
return resultSb.String()
case *ReferenceLink:
var resultSb strings.Builder
resultSb.WriteString(`<a href="` + htmlEscaper.Replace(escapeURL(v.Destination())) + `"`)
if title := v.Title(); title != "" {
resultSb.WriteString(` title="` + htmlEscaper.Replace(title) + `"`)
}
resultSb.WriteString(`>`)
for _, inline := range v.Children {
resultSb.WriteString(RenderInlineHTML(inline))
}
resultSb.WriteString("</a>")
return resultSb.String()
case *Autolink:
var resultSb strings.Builder
resultSb.WriteString(`<a href="` + htmlEscaper.Replace(escapeURL(v.Destination())) + `">`)
for _, inline := range v.Children {
resultSb.WriteString(RenderInlineHTML(inline))
}
resultSb.WriteString("</a>")
return resultSb.String()
case *Emoji:
escapedName := htmlEscaper.Replace(v.Name)
result += fmt.Sprintf(`<span data-emoji-name="%s" data-literal=":%s:" />`, escapedName, escapedName)
default:
panic(fmt.Sprintf("missing case for type %T", v))
}
return
}
func renderImageAltText(children []Inline) string {
var resultSb strings.Builder
for _, inline := range children {
resultSb.WriteString(renderImageChildAltText(inline))
}
return resultSb.String()
}
func renderImageChildAltText(inline Inline) string {
switch v := inline.(type) {
case *Text:
return v.Text
case *InlineImage:
var resultSb strings.Builder
for _, inline := range v.Children {
resultSb.WriteString(renderImageChildAltText(inline))
}
return resultSb.String()
case *InlineLink:
var resultSb strings.Builder
for _, inline := range v.Children {
resultSb.WriteString(renderImageChildAltText(inline))
}
return resultSb.String()
}
return ""
}