package main import ( "bytes" "errors" "fmt" "io/ioutil" "os" "path/filepath" "regexp" "github.com/acstech/liquid" "github.com/russross/blackfriday" yaml "gopkg.in/yaml.v2" ) const ( printFrontmatter = false ) var ( frontmatterMatcher = regexp.MustCompile(`(?s)^---\n(.+?\n)---\n`) templateVariableMatcher = regexp.MustCompile(`:(?:collection|file_ext|name|path|title)\b`) nonAlphanumericSequenceMatcher = regexp.MustCompile(`[^[:alnum:]]+`) ) var permalinkStyles = map[string]string{ "date": "/:categories/:year/:month/:day/:title.html", "pretty": "/:categories/:year/:month/:day/:title/", "ordinal": "/:categories/:year/:y_day/:title.html", "none": "/:categories/:title.html", } // A Page represents an HTML page. type Page struct { Path string Permalink string Static bool Published bool FrontMatter map[interface{}]interface{} Content []byte } func (p Page) String() string { return fmt.Sprintf("Page{Path=%v, Permalink=%v, Static=%v}", p.Path, p.Permalink, p.Static) } // CollectionItemData returns metadata for use in the representation of the page as a collection item func (p Page) CollectionItemData() map[interface{}]interface{} { // should have title, parts, url, description, due_date data := map[interface{}]interface{}{ "url": p.Permalink, } // TODO additional variables from https://jekyllrb.com/docs/collections/#documents if p.FrontMatter != nil { data = mergeMaps(data, p.FrontMatter) } return data } func readPage(path string, defaults map[interface{}]interface{}) (*Page, error) { var ( frontMatter map[interface{}]interface{} static = true body []byte ) // TODO don't read, parse binary files source, err := ioutil.ReadFile(filepath.Join(siteConfig.SourceDir, path)) if err != nil { return nil, err } data := defaults if match := frontmatterMatcher.FindSubmatchIndex(source); match != nil { static = false // TODO only prepend newlines if it's markdown body = append( regexp.MustCompile(`[^\n\r]+`).ReplaceAllLiteral(source[:match[1]], []byte{}), source[match[1]:]...) frontMatter = map[interface{}]interface{}{} err = yaml.Unmarshal(source[match[2]:match[3]], &frontMatter) if err != nil { err := &os.PathError{Op: "read frontmatter", Path: path, Err: err} return nil, err } data = mergeMaps(data, frontMatter) } else { body = []byte{} } permalink := path if val, ok := data["permalink"]; ok { pattern, ok := val.(string) if !ok { err := errors.New("permalink value must be a string") err = &os.PathError{Op: "render", Path: path, Err: err} return nil, err } permalink = expandPermalinkPattern(pattern, data, path) } return &Page{ Path: path, Permalink: permalink, Static: static, Published: getBool(data, "published", true), FrontMatter: frontMatter, Content: body, }, nil } // Render applies Liquid and Markdown, as appropriate. func (p Page) Render() ([]byte, error) { var ( path = p.Path ext = filepath.Ext(path) data = p.FrontMatter ) if printFrontmatter { b, _ := yaml.Marshal(stringMap(data)) println(string(b)) } template, err := liquid.Parse(p.Content, nil) if err != nil { err := &os.PathError{Op: "Liquid Error", Path: path, Err: err} return nil, err } writer := new(bytes.Buffer) template.Render(writer, stringMap(data)) body := writer.Bytes() if ext == ".md" { body = blackfriday.MarkdownBasic(body) } return body, nil } func expandPermalinkPattern(pattern string, data map[interface{}]interface{}, path string) string { if p, found := permalinkStyles[pattern]; found { pattern = p } var ( collectionName string ext = filepath.Ext(path) localPath = path outputExt = ext name = filepath.Base(localPath) title = getString(data, "title", name[:len(name)-len(ext)]) ) if ext == ".md" { outputExt = "" localPath = localPath[:len(localPath)-len(ext)] } if val, found := data["collection"]; found { collectionName = val.(string) collectionPath := "_" + collectionName + "/" localPath = localPath[len(collectionPath):] } replaceNonalphumericsByHyphens := func(s string) string { return nonAlphanumericSequenceMatcher.ReplaceAllString(s, "-") } templateVariables := map[string]string{ "collection": collectionName, "name": replaceNonalphumericsByHyphens(name), "output_ext": outputExt, "path": localPath, "title": replaceNonalphumericsByHyphens(title), // TODO year month imonth day i_day short_year hour minute second slug categories } return templateVariableMatcher.ReplaceAllStringFunc(pattern, func(m string) string { varname := m[1:] value := templateVariables[varname] if value == "" { fmt.Printf("unknown variable %s in permalink template\n", varname) } return value }) }