mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-30 18:22:49 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			209 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			209 lines
		
	
	
		
			5.4 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
 | |
| 	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>`
 | |
| )
 |