all: Add org-mode support
authorChase Adams <realchaseadams@gmail.com>
Tue, 21 Feb 2017 07:46:03 +0000 (23:46 -0800)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Tue, 21 Feb 2017 07:46:03 +0000 (08:46 +0100)
Fixes #1483
See #936

docs/content/content/summaries.md
docs/content/content/supported-formats.md
helpers/content.go
helpers/general.go
helpers/general_test.go
hugolib/handler_page.go
hugolib/page.go
parser/frontmatter.go
parser/page.go
vendor/vendor.json

index 0f2e6433bee87eb8b008a7d7b7f8435dc3a1cf44..1f65d431df515a37bb9e3f9b53d3db1144cdfaef 100644 (file)
@@ -24,14 +24,14 @@ By default, Hugo automatically takes the first 70 words of your content as its s
 
 ## User-defined: manual summary split:
 
-Alternatively, you may add the <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</code> summary divider[^1] where you want to split the article.  Content prior to the summary divider will be used as that content's summary, and stored into the `.Summary` variable with all HTML formatting intact.
+Alternatively, you may add the <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</code> summary divider[^1] (for org content, use <code># more</code>) where you want to split the article.  Content prior to the summary divider will be used as that content's summary, and stored into the `.Summary` variable with all HTML formatting intact.
 
 [^1]: The **summary divider** is also called "more tag", "excerpt separator", etc. in other literature.
 
 * Pros: Freedom, precision, and improved rendering.  All formatting is preserved.
-* Cons: Need to remember to type <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</code> in your content file.  :-)
+* Cons: Need to remember to type <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</code> (or <code># more</code> for org content) in your content file.  :-)
 
-Be careful to enter <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</code> exactly, i.e. all lowercase with no whitespace, otherwise it would be treated as regular comment and ignored.
+Be careful to enter <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</code> (or <code># more</code> for org content) exactly, i.e. all lowercase with no whitespace, otherwise it would be treated as regular comment and ignored.
 
 If there is nothing but spaces and newlines after the summary divider then `.Truncated` will be false.
 
index 68aa66acc95c65148034cf986f18e6da91405c4d..1b30335964d601c16b8b4e7ebeb2014b42197e44 100644 (file)
@@ -13,7 +13,7 @@ weight: 15
 toc: true
 ---
 
-  Since 0.14, Hugo has defined a new concept called _external helpers_. It means that you can write your content using Asciidoc[tor], or reStructuredText. If you have files with associated extensions ([details](https://github.com/spf13/hugo/blob/77c60a3440806067109347d04eb5368b65ea0fe8/helpers/general.go#L65)), then Hugo will call external commands to generate the content.
+  Since 0.14, Hugo has defined a new concept called _external helpers_. It means that you can write your content using Asciidoc[tor], reStructuredText or Org-Mode. If you have files with associated extensions ([details](https://github.com/spf13/hugo/blob/77c60a3440806067109347d04eb5368b65ea0fe8/helpers/general.go#L65)), then Hugo will call external commands to generate the content (the exception being Org-Mode content, which is parsed natively).
 
   This means that you will have to install the associated tool on your machine to be able to use those formats.
 
index 0eb687af6a7880f8c4b063ce4b65a5d5bf361095..7b529304d4c0e35ef806b86b7303e3e95b525ebf 100644 (file)
@@ -24,12 +24,12 @@ import (
        "unicode"
        "unicode/utf8"
 
-       "github.com/spf13/hugo/config"
-
+       "github.com/chaseadamsio/goorgeous"
        "github.com/miekg/mmark"
        "github.com/mitchellh/mapstructure"
        "github.com/russross/blackfriday"
        bp "github.com/spf13/hugo/bufferpool"
+       "github.com/spf13/hugo/config"
        jww "github.com/spf13/jwalterweatherman"
 
        "strings"
@@ -415,6 +415,8 @@ func (c ContentSpec) RenderBytes(ctx *RenderingContext) []byte {
                return c.mmarkRender(ctx)
        case "rst":
                return getRstContent(ctx)
+       case "org":
+               return orgRender(ctx, c)
        }
 }
 
@@ -663,3 +665,10 @@ func getRstContent(ctx *RenderingContext) []byte {
 
        return result[bodyStart+7 : bodyEnd]
 }
+
+func orgRender(ctx *RenderingContext, c ContentSpec) []byte {
+       content := ctx.Content
+       cleanContent := bytes.Replace(content, []byte("# more"), []byte(""), 1)
+       return goorgeous.Org(cleanContent,
+               c.getHTMLRenderer(blackfriday.HTML_TOC, ctx))
+}
index 91c25c04caefbee24f0de64faebcfe6d940679a0..f9e67e72d72913361d6054d7acc3995b03f69232 100644 (file)
@@ -80,6 +80,8 @@ func GuessType(in string) string {
                return "rst"
        case "html", "htm":
                return "html"
+       case "org":
+               return "org"
        }
 
        return "unknown"
index 8afdb59eb4163446177b0182211ad0c2ed0c9ff1..3fa587e78b212fe426c40536367a81309c94a1d9 100644 (file)
@@ -36,6 +36,7 @@ func TestGuessType(t *testing.T) {
                {"mmark", "mmark"},
                {"html", "html"},
                {"htm", "html"},
+               {"org", "org"},
                {"excel", "unknown"},
        } {
                result := GuessType(this.in)
index a47758f84753d0d1d146a7aa0cb725af0afd1070..6439344b055a75a3e9b6c340f48116dbe2b3660b 100644 (file)
@@ -26,6 +26,7 @@ func init() {
        RegisterHandler(new(asciidocHandler))
        RegisterHandler(new(rstHandler))
        RegisterHandler(new(mmarkHandler))
+       RegisterHandler(new(orgHandler))
 }
 
 type basicPageHandler Handle
@@ -110,6 +111,15 @@ func (h mmarkHandler) PageConvert(p *Page) HandledResult {
        return commonConvert(p)
 }
 
+type orgHandler struct {
+       basicPageHandler
+}
+
+func (h orgHandler) Extensions() []string { return []string{"org"} }
+func (h orgHandler) PageConvert(p *Page) HandledResult {
+       return commonConvert(p)
+}
+
 func commonConvert(p *Page) HandledResult {
        if p.rendered {
                panic(fmt.Sprintf("Page %q already rendered, does not need conversion", p.BaseFileName()))
index 23e11e5b4770bf58ef0532b7b31d45cb21fb5f0a..07d073c9e59861d08b4ac4ba9a5f9790941640a0 100644 (file)
@@ -420,7 +420,11 @@ var (
 // rendering engines.
 // TODO(bep) inline replace
 func (p *Page) replaceDivider(content []byte) []byte {
-       sections := bytes.Split(content, helpers.SummaryDivider)
+       summaryDivider := helpers.SummaryDivider
+       if p.Ext() == "org" {
+               summaryDivider = []byte("# more")
+       }
+       sections := bytes.Split(content, summaryDivider)
 
        // If the raw content has nothing but whitespace after the summary
        // marker then the page shouldn't be marked as truncated.  This check
index ed25c08e1284dfba8004e77e6b6a7de741d62340..e57a593ab4ec8f44beff9bb15dc5d04dcf924d69 100644 (file)
@@ -19,6 +19,7 @@ import (
        "errors"
        "strings"
 
+       "github.com/chaseadamsio/goorgeous"
        toml "github.com/pelletier/go-toml"
 
        "gopkg.in/yaml.v2"
@@ -154,6 +155,8 @@ func DetectFrontMatter(mark rune) (f *frontmatterType) {
                return &frontmatterType{[]byte(TOMLDelim), []byte(TOMLDelim), HandleTOMLMetaData, false}
        case '{':
                return &frontmatterType{[]byte{'{'}, []byte{'}'}, HandleJSONMetaData, true}
+       case '#':
+               return &frontmatterType{[]byte("#+"), []byte("\n"), HandleOrgMetaData, false}
        default:
                return nil
        }
@@ -189,3 +192,7 @@ func HandleJSONMetaData(datum []byte) (interface{}, error) {
        err := json.Unmarshal(datum, &f)
        return f, err
 }
+
+func HandleOrgMetaData(datum []byte) (interface{}, error) {
+       return goorgeous.OrgHeaders(datum)
+}
index 57b2d30450fc03a7fb1be40690921688a1865317..3347380d7f37ca624aeabc943f73c8b1e3e812a6 100644 (file)
@@ -21,6 +21,8 @@ import (
        "regexp"
        "strings"
        "unicode"
+
+       "github.com/chaseadamsio/goorgeous"
 )
 
 const (
@@ -91,9 +93,11 @@ func (p *page) Metadata() (meta interface{}, err error) {
 
        if len(frontmatter) != 0 {
                fm := DetectFrontMatter(rune(frontmatter[0]))
-               meta, err = fm.Parse(frontmatter)
-               if err != nil {
-                       return
+               if fm != nil {
+                       meta, err = fm.Parse(frontmatter)
+                       if err != nil {
+                               return
+                       }
                }
        }
        return
@@ -129,6 +133,12 @@ func ReadFrom(r io.Reader) (p Page, err error) {
                        return nil, err
                }
                newp.frontmatter = fm
+       } else if newp.render && goorgeous.IsKeyword(firstLine) {
+               fm, err := goorgeous.ExtractOrgHeaders(reader)
+               if err != nil {
+                       return nil, err
+               }
+               newp.frontmatter = fm
        }
 
        content, err := extractContent(reader)
index 95641d00086b62651ee4730bccc330611ab4e194..8749dee0d186bbf14d38937e7fc77bf3598c3f26 100644 (file)
                        "revision": "b896c45f5af983b1f416bdf3bb89c4f1f0926f69",
                        "revisionTime": "2016-04-08T19:03:23Z"
                },
+               {
+                       "checksumSHA1": "RxIwAgjIuBpwde5BCZRLLK7VRG8=",
+                       "path": "github.com/chaseadamsio/goorgeous",
+                       "revision": "72a06e1b07db57f3931f5a9c00f3f04e636ad0a8",
+                       "revisionTime": "2017-02-17T13:03:04Z"
+               },
                {
                        "checksumSHA1": "ntacCkWfMT63DaehXLG5FeXWyNM=",
                        "path": "github.com/cpuguy83/go-md2man/md2man",