| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- // Copyright 2017 The Gitea Authors. All rights reserved.
- // SPDX-License-Identifier: MIT
-
- package orgmode
-
- import (
- "fmt"
- "html"
- "html/template"
- "io"
- "strings"
-
- "code.gitea.io/gitea/modules/highlight"
- "code.gitea.io/gitea/modules/htmlutil"
- "code.gitea.io/gitea/modules/log"
- "code.gitea.io/gitea/modules/markup"
- "code.gitea.io/gitea/modules/setting"
-
- "github.com/alecthomas/chroma/v2"
- "github.com/alecthomas/chroma/v2/lexers"
- "github.com/niklasfasching/go-org/org"
- )
-
- func init() {
- markup.RegisterRenderer(renderer{})
- }
-
- // Renderer implements markup.Renderer for orgmode
- type renderer struct{}
-
- var (
- _ markup.Renderer = (*renderer)(nil)
- _ markup.PostProcessRenderer = (*renderer)(nil)
- )
-
- // Name implements markup.Renderer
- func (renderer) Name() string {
- return "orgmode"
- }
-
- // NeedPostProcess implements markup.PostProcessRenderer
- func (renderer) NeedPostProcess() bool { return true }
-
- // Extensions implements markup.Renderer
- func (renderer) Extensions() []string {
- return []string{".org"}
- }
-
- // SanitizerRules implements markup.Renderer
- func (renderer) SanitizerRules() []setting.MarkupSanitizerRule {
- return []setting.MarkupSanitizerRule{}
- }
-
- // Render renders orgmode raw bytes to HTML
- func Render(ctx *markup.RenderContext, input io.Reader, output io.Writer) error {
- htmlWriter := org.NewHTMLWriter()
- htmlWriter.HighlightCodeBlock = func(source, lang string, inline bool, params map[string]string) string {
- defer func() {
- if err := recover(); err != nil {
- log.Error("Panic in HighlightCodeBlock: %v\n%s", err, log.Stack(2))
- panic(err)
- }
- }()
- w := &strings.Builder{}
-
- lexer := lexers.Get(lang)
- if lexer == nil && lang == "" {
- lexer = lexers.Analyse(source)
- if lexer == nil {
- lexer = lexers.Fallback
- }
- lang = strings.ToLower(lexer.Config().Name)
- }
-
- // include language-x class as part of commonmark spec
- if err := ctx.RenderInternal.FormatWithSafeAttrs(w, `<pre><code class="chroma language-%s">`, lang); err != nil {
- return ""
- }
- if lexer == nil {
- if _, err := w.WriteString(html.EscapeString(source)); err != nil {
- return ""
- }
- } else {
- lexer = chroma.Coalesce(lexer)
- if _, err := w.WriteString(string(highlight.CodeFromLexer(lexer, source))); err != nil {
- return ""
- }
- }
- if _, err := w.WriteString("</code></pre>"); err != nil {
- return ""
- }
-
- return w.String()
- }
-
- w := &orgWriter{rctx: ctx, HTMLWriter: htmlWriter}
- htmlWriter.ExtendingWriter = w
-
- res, err := org.New().Silent().Parse(input, "").Write(w)
- if err != nil {
- return fmt.Errorf("orgmode.Render failed: %w", err)
- }
- _, err = io.Copy(output, strings.NewReader(res))
- return err
- }
-
- // RenderString renders orgmode string to HTML string
- func RenderString(ctx *markup.RenderContext, content string) (string, error) {
- var buf strings.Builder
- if err := Render(ctx, strings.NewReader(content), &buf); err != nil {
- return "", err
- }
- return buf.String(), nil
- }
-
- // Render renders orgmode string to HTML string
- func (renderer) Render(ctx *markup.RenderContext, input io.Reader, output io.Writer) error {
- return Render(ctx, input, output)
- }
-
- type orgWriter struct {
- *org.HTMLWriter
- rctx *markup.RenderContext
- }
-
- var _ org.Writer = (*orgWriter)(nil)
-
- func (r *orgWriter) resolveLink(link string) string {
- return strings.TrimPrefix(link, "file:")
- }
-
- // WriteRegularLink renders images, links or videos
- func (r *orgWriter) WriteRegularLink(l org.RegularLink) {
- link := r.resolveLink(l.URL)
-
- printHTML := func(html template.HTML, a ...any) {
- _, _ = fmt.Fprint(r, htmlutil.HTMLFormat(html, a...))
- }
- // Inspired by https://github.com/niklasfasching/go-org/blob/6eb20dbda93cb88c3503f7508dc78cbbc639378f/org/html_writer.go#L406-L427
- switch l.Kind() {
- case "image":
- if l.Description == nil {
- printHTML(`<img src="%s" alt="%s">`, link, link)
- } else {
- imageSrc := r.resolveLink(org.String(l.Description...))
- printHTML(`<a href="%s"><img src="%s" alt="%s"></a>`, link, imageSrc, imageSrc)
- }
- case "video":
- if l.Description == nil {
- printHTML(`<video src="%s">%s</video>`, link, link)
- } else {
- videoSrc := r.resolveLink(org.String(l.Description...))
- printHTML(`<a href="%s"><video src="%s">%s</video></a>`, link, videoSrc, videoSrc)
- }
- default:
- var description any = link
- if l.Description != nil {
- description = template.HTML(r.WriteNodesAsString(l.Description...)) // orgmode HTMLWriter outputs HTML content
- }
- printHTML(`<a href="%s">%s</a>`, link, description)
- }
- }
|