mirror of
https://github.com/thomiceli/opengist.git
synced 2025-01-05 17:02:39 +00:00
188 lines
3.9 KiB
Go
188 lines
3.9 KiB
Go
package render
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"encoding/base64"
|
|
"fmt"
|
|
"github.com/alecthomas/chroma/v2"
|
|
"github.com/alecthomas/chroma/v2/formatters/html"
|
|
"github.com/alecthomas/chroma/v2/lexers"
|
|
"github.com/alecthomas/chroma/v2/styles"
|
|
"github.com/rs/zerolog/log"
|
|
"github.com/thomiceli/opengist/internal/db"
|
|
"github.com/thomiceli/opengist/internal/git"
|
|
"path"
|
|
"sync"
|
|
)
|
|
|
|
type RenderedFile struct {
|
|
*git.File
|
|
Type string `json:"type"`
|
|
Lines []string `json:"-"`
|
|
HTML string `json:"-"`
|
|
}
|
|
|
|
type RenderedGist struct {
|
|
*db.Gist
|
|
Lines []string
|
|
HTML string
|
|
}
|
|
|
|
func HighlightFile(file *git.File) (RenderedFile, error) {
|
|
style := newStyle()
|
|
lexer := newLexer(file.Filename)
|
|
|
|
if lexer.Config().Name == "markdown" {
|
|
return MarkdownFile(file)
|
|
}
|
|
if lexer.Config().Name == "XML" && path.Ext(file.Filename) == ".svg" {
|
|
return RenderSvgFile(file), nil
|
|
}
|
|
|
|
formatter := html.New(html.WithClasses(true), html.PreventSurroundingPre(true))
|
|
|
|
rendered := RenderedFile{
|
|
File: file,
|
|
}
|
|
|
|
iterator, err := lexer.Tokenise(nil, file.Content+"\n")
|
|
if err != nil {
|
|
return rendered, err
|
|
}
|
|
|
|
htmlbuf := bytes.Buffer{}
|
|
w := bufio.NewWriter(&htmlbuf)
|
|
|
|
tokensLines := chroma.SplitTokensIntoLines(iterator.Tokens())
|
|
lines := make([]string, 0, len(tokensLines))
|
|
for _, tokens := range tokensLines {
|
|
iterator = chroma.Literator(tokens...)
|
|
err = formatter.Format(&htmlbuf, style, iterator)
|
|
if err != nil {
|
|
return rendered, fmt.Errorf("unable to format code: %w", err)
|
|
}
|
|
lines = append(lines, htmlbuf.String())
|
|
htmlbuf.Reset()
|
|
}
|
|
|
|
_ = w.Flush()
|
|
|
|
rendered.Lines = lines
|
|
rendered.Type = parseFileTypeName(*lexer.Config())
|
|
|
|
return rendered, err
|
|
}
|
|
|
|
func HighlightFiles(files []*git.File) []RenderedFile {
|
|
const numWorkers = 10
|
|
jobs := make(chan int, numWorkers)
|
|
renderedFiles := make([]RenderedFile, len(files))
|
|
var wg sync.WaitGroup
|
|
|
|
worker := func() {
|
|
for idx := range jobs {
|
|
rendered, err := HighlightFile(files[idx])
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Error rendering gist preview for " + files[idx].Filename)
|
|
}
|
|
renderedFiles[idx] = rendered
|
|
}
|
|
wg.Done()
|
|
}
|
|
|
|
for i := 0; i < numWorkers; i++ {
|
|
wg.Add(1)
|
|
go worker()
|
|
}
|
|
|
|
for i := range files {
|
|
jobs <- i
|
|
}
|
|
close(jobs)
|
|
|
|
wg.Wait()
|
|
|
|
return renderedFiles
|
|
}
|
|
|
|
func HighlightGistPreview(gist *db.Gist) (RenderedGist, error) {
|
|
rendered := RenderedGist{
|
|
Gist: gist,
|
|
}
|
|
|
|
style := newStyle()
|
|
lexer := newLexer(gist.PreviewFilename)
|
|
if lexer.Config().Name == "markdown" {
|
|
return MarkdownGistPreview(gist)
|
|
}
|
|
|
|
formatter := html.New(html.WithClasses(true), html.PreventSurroundingPre(true))
|
|
|
|
iterator, err := lexer.Tokenise(nil, gist.Preview)
|
|
if err != nil {
|
|
return rendered, err
|
|
}
|
|
|
|
htmlbuf := bytes.Buffer{}
|
|
w := bufio.NewWriter(&htmlbuf)
|
|
|
|
tokensLines := chroma.SplitTokensIntoLines(iterator.Tokens())
|
|
lines := make([]string, 0, len(tokensLines))
|
|
for _, tokens := range tokensLines {
|
|
iterator = chroma.Literator(tokens...)
|
|
err = formatter.Format(&htmlbuf, style, iterator)
|
|
if err != nil {
|
|
return rendered, fmt.Errorf("unable to format code: %w", err)
|
|
}
|
|
lines = append(lines, htmlbuf.String())
|
|
htmlbuf.Reset()
|
|
}
|
|
|
|
_ = w.Flush()
|
|
|
|
rendered.Lines = lines
|
|
|
|
return rendered, err
|
|
}
|
|
|
|
func RenderSvgFile(file *git.File) RenderedFile {
|
|
rendered := RenderedFile{
|
|
File: file,
|
|
}
|
|
|
|
encoded := base64.StdEncoding.EncodeToString([]byte(file.Content))
|
|
content := `<img src="data:image/svg+xml;base64,` + encoded + `" />`
|
|
|
|
rendered.HTML = content
|
|
rendered.Type = "SVG"
|
|
|
|
return rendered
|
|
}
|
|
|
|
func parseFileTypeName(config chroma.Config) string {
|
|
fileType := config.Name
|
|
if fileType == "fallback" || fileType == "plaintext" {
|
|
return "Text"
|
|
}
|
|
|
|
return fileType
|
|
}
|
|
|
|
func newLexer(filename string) chroma.Lexer {
|
|
var lexer chroma.Lexer
|
|
if lexer = lexers.Get(filename); lexer == nil {
|
|
lexer = lexers.Fallback
|
|
}
|
|
|
|
return lexer
|
|
}
|
|
|
|
func newStyle() *chroma.Style {
|
|
var style *chroma.Style
|
|
if style = styles.Get("catppuccin-latte"); style == nil {
|
|
style = styles.Fallback
|
|
}
|
|
|
|
return style
|
|
}
|