mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-11-03 19:17:29 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			209 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			209 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package markdown
 | 
						|
 | 
						|
import (
 | 
						|
	"bytes"
 | 
						|
	"io/ioutil"
 | 
						|
	"log"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"strings"
 | 
						|
	"text/template"
 | 
						|
 | 
						|
	"github.com/mholt/caddy/middleware"
 | 
						|
	"github.com/russross/blackfriday"
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	// DefaultTemplate is the default template.
 | 
						|
	DefaultTemplate = "defaultTemplate"
 | 
						|
	// DefaultStaticDir is the default static directory.
 | 
						|
	DefaultStaticDir = "generated_site"
 | 
						|
)
 | 
						|
 | 
						|
// Data represents a markdown document.
 | 
						|
type Data struct {
 | 
						|
	middleware.Context
 | 
						|
	Doc   map[string]string
 | 
						|
	Links []PageLink
 | 
						|
}
 | 
						|
 | 
						|
// Process processes the contents of a page in b. It parses the metadata
 | 
						|
// (if any) and uses the template (if found).
 | 
						|
func (md Markdown) Process(c *Config, requestPath string, b []byte, ctx middleware.Context) ([]byte, error) {
 | 
						|
	var metadata = Metadata{Variables: make(map[string]string)}
 | 
						|
	var markdown []byte
 | 
						|
	var err error
 | 
						|
 | 
						|
	// find parser compatible with page contents
 | 
						|
	parser := findParser(b)
 | 
						|
 | 
						|
	if parser == nil {
 | 
						|
		// if not found, assume whole file is markdown (no front matter)
 | 
						|
		markdown = b
 | 
						|
	} else {
 | 
						|
		// if found, assume metadata present and parse.
 | 
						|
		markdown, err = parser.Parse(b)
 | 
						|
		if err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
		metadata = parser.Metadata()
 | 
						|
	}
 | 
						|
 | 
						|
	// if template is not specified, check if Default template is set
 | 
						|
	if metadata.Template == "" {
 | 
						|
		if _, ok := c.Templates[DefaultTemplate]; ok {
 | 
						|
			metadata.Template = DefaultTemplate
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// if template is set, load it
 | 
						|
	var tmpl []byte
 | 
						|
	if metadata.Template != "" {
 | 
						|
		if t, ok := c.Templates[metadata.Template]; ok {
 | 
						|
			tmpl, err = ioutil.ReadFile(t)
 | 
						|
		}
 | 
						|
		if err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// process markdown
 | 
						|
	extns := blackfriday.EXTENSION_TABLES | blackfriday.EXTENSION_FENCED_CODE | blackfriday.EXTENSION_STRIKETHROUGH | blackfriday.EXTENSION_DEFINITION_LISTS
 | 
						|
	markdown = blackfriday.Markdown(markdown, c.Renderer, extns)
 | 
						|
 | 
						|
	// set it as body for template
 | 
						|
	metadata.Variables["body"] = string(markdown)
 | 
						|
	title := metadata.Title
 | 
						|
	if title == "" {
 | 
						|
		title = filepath.Base(requestPath)
 | 
						|
		var extension = filepath.Ext(requestPath)
 | 
						|
		title = title[0 : len(title)-len(extension)]
 | 
						|
	}
 | 
						|
	metadata.Variables["title"] = title
 | 
						|
 | 
						|
	return md.processTemplate(c, requestPath, tmpl, metadata, ctx)
 | 
						|
}
 | 
						|
 | 
						|
// processTemplate processes a template given a requestPath,
 | 
						|
// template (tmpl) and metadata
 | 
						|
func (md Markdown) processTemplate(c *Config, requestPath string, tmpl []byte, metadata Metadata, ctx middleware.Context) ([]byte, error) {
 | 
						|
	// if template is not specified,
 | 
						|
	// use the default template
 | 
						|
	if tmpl == nil {
 | 
						|
		tmpl = defaultTemplate(c, metadata, requestPath)
 | 
						|
	}
 | 
						|
 | 
						|
	// process the template
 | 
						|
	b := new(bytes.Buffer)
 | 
						|
	t, err := template.New("").Parse(string(tmpl))
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	mdData := Data{
 | 
						|
		Context: ctx,
 | 
						|
		Doc:     metadata.Variables,
 | 
						|
		Links:   c.Links,
 | 
						|
	}
 | 
						|
 | 
						|
	c.RLock()
 | 
						|
	err = t.Execute(b, mdData)
 | 
						|
	c.RUnlock()
 | 
						|
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	// generate static page
 | 
						|
	if err = md.generatePage(c, requestPath, b.Bytes()); err != nil {
 | 
						|
		// if static page generation fails, nothing fatal, only log the error.
 | 
						|
		// TODO: Report (return) this non-fatal error, but don't log it here?
 | 
						|
		log.Println("[ERROR] markdown: Render:", err)
 | 
						|
	}
 | 
						|
 | 
						|
	return b.Bytes(), nil
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
// generatePage generates a static html page from the markdown in content if c.StaticDir
 | 
						|
// is a non-empty value, meaning that the user enabled static site generation.
 | 
						|
func (md Markdown) generatePage(c *Config, requestPath string, content []byte) error {
 | 
						|
	// Only generate the page if static site generation is enabled
 | 
						|
	if c.StaticDir != "" {
 | 
						|
		// if static directory is not existing, create it
 | 
						|
		if _, err := os.Stat(c.StaticDir); err != nil {
 | 
						|
			err := os.MkdirAll(c.StaticDir, os.FileMode(0755))
 | 
						|
			if err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		// the URL will always use "/" as a path separator,
 | 
						|
		// convert that to a native path to support OS that
 | 
						|
		// use different path separators
 | 
						|
		filePath := filepath.Join(c.StaticDir, filepath.FromSlash(requestPath))
 | 
						|
 | 
						|
		// If it is index file, use the directory instead
 | 
						|
		if md.IsIndexFile(filepath.Base(requestPath)) {
 | 
						|
			filePath, _ = filepath.Split(filePath)
 | 
						|
		}
 | 
						|
 | 
						|
		// Create the directory in case it is not existing
 | 
						|
		if err := os.MkdirAll(filePath, os.FileMode(0744)); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		// generate index.html file in the directory
 | 
						|
		filePath = filepath.Join(filePath, "index.html")
 | 
						|
		err := ioutil.WriteFile(filePath, content, os.FileMode(0664))
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		c.Lock()
 | 
						|
		c.StaticFiles[requestPath] = filepath.ToSlash(filePath)
 | 
						|
		c.Unlock()
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// defaultTemplate constructs a default template.
 | 
						|
func defaultTemplate(c *Config, metadata Metadata, requestPath string) []byte {
 | 
						|
	var scripts, styles bytes.Buffer
 | 
						|
	for _, style := range c.Styles {
 | 
						|
		styles.WriteString(strings.Replace(cssTemplate, "{{url}}", style, 1))
 | 
						|
		styles.WriteString("\r\n")
 | 
						|
	}
 | 
						|
	for _, script := range c.Scripts {
 | 
						|
		scripts.WriteString(strings.Replace(jsTemplate, "{{url}}", script, 1))
 | 
						|
		scripts.WriteString("\r\n")
 | 
						|
	}
 | 
						|
 | 
						|
	// Title is first line (length-limited), otherwise filename
 | 
						|
	title, _ := metadata.Variables["title"]
 | 
						|
 | 
						|
	html := []byte(htmlTemplate)
 | 
						|
	html = bytes.Replace(html, []byte("{{title}}"), []byte(title), 1)
 | 
						|
	html = bytes.Replace(html, []byte("{{css}}"), styles.Bytes(), 1)
 | 
						|
	html = bytes.Replace(html, []byte("{{js}}"), scripts.Bytes(), 1)
 | 
						|
 | 
						|
	return html
 | 
						|
}
 | 
						|
 | 
						|
const (
 | 
						|
	htmlTemplate = `<!DOCTYPE html>
 | 
						|
<html>
 | 
						|
	<head>
 | 
						|
		<title>{{title}}</title>
 | 
						|
		<meta charset="utf-8">
 | 
						|
		{{css}}
 | 
						|
		{{js}}
 | 
						|
	</head>
 | 
						|
	<body>
 | 
						|
		{{.Doc.body}}
 | 
						|
	</body>
 | 
						|
</html>`
 | 
						|
	cssTemplate = `<link rel="stylesheet" href="{{url}}">`
 | 
						|
	jsTemplate  = `<script src="{{url}}"></script>`
 | 
						|
)
 |