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 = ` {{title}} {{css}} {{js}} {{.Doc.body}} ` cssTemplate = `` jsTemplate = `` )