From 358dcce7a6b4a8100964597f81cf371451f6c4d3 Mon Sep 17 00:00:00 2001 From: Anthony Fok Date: Fri, 23 Jan 2015 11:59:14 -0700 Subject: [PATCH] Experimental AsciiDoc support with external helpers See #470 * Based on existing support for reStructuredText files * Handles content files with extensions `.asciidoc` and `.ad` * Pipes content through `asciidoctor --safe -`. If `asciidoctor` is not installed, then `asciidoc --safe -`. * To make sure `asciidoctor` or `asciidoc` is found, after adding a piece of AsciiDoc content, run `hugo` with the `-v` flag and look for this message: INFO: 2015/01/23 Rendering with /usr/bin/asciidoctor ... Caveats: * The final "Last updated" timestamp is currently not stripped. * When `hugo` is run with `-v`, you may see a lot of these messages INFO: 2015/01/23 Rendering with /usr/bin/asciidoctor ... if you have lots of `*.ad`, `*.adoc` or `*.asciidoc` files. * Some versions of `asciidoc` may have trouble with its safe mode. To test if you are affected, try this: $ echo "Hello" | asciidoc --safe - asciidoc: ERROR: unsafe: ifeval invalid asciidoc: FAILED: ifeval invalid safe document If so, I recommend that you install `asciidoctor` instead. Feedback and patches welcome! Ideally, we should be using https://github.com/VonC/asciidocgo, @VonC's wonderful Go implementation of Asciidoctor. However, there is still a bit of work needed for asciidocgo to expose its API so that Hugo can actually use it. Until then, hope this "experimental AsciiDoc support through external helpers" can serve as a stopgap solution for our community. :-) 2015-01-30: Updated for the replaceShortcodeTokens() syntax change 2015-02-21: Add `.adoc` extension as suggested by @Fale Conflicts: helpers/content.go --- helpers/content.go | 37 +++++++++++++++++++++++++++++++++++++ helpers/general.go | 2 ++ helpers/general_test.go | 3 +++ hugolib/handler_page.go | 17 ++++++++++++++++- 4 files changed, 58 insertions(+), 1 deletion(-) diff --git a/helpers/content.go b/helpers/content.go index a327a1e5..daf7e267 100644 --- a/helpers/content.go +++ b/helpers/content.go @@ -230,6 +230,8 @@ func RenderBytesWithTOC(ctx *RenderingContext) []byte { return markdownRenderWithTOC(ctx) case "markdown": return markdownRenderWithTOC(ctx) + case "asciidoc": + return []byte(GetAsciidocContent(ctx.Content)) case "rst": return []byte(GetRstContent(ctx.Content)) } @@ -242,6 +244,8 @@ func RenderBytes(ctx *RenderingContext) []byte { return markdownRender(ctx) case "markdown": return markdownRender(ctx) + case "asciidoc": + return []byte(GetAsciidocContent(ctx.Content)) case "rst": return []byte(GetRstContent(ctx.Content)) } @@ -299,6 +303,39 @@ func TruncateWordsToWholeSentence(words []string, max int) (string, bool) { return strings.Join(words[:max], " "), true } +// GetAsciidocContent calls asciidoctor or asciidoc as an external helper +// to convert AsciiDoc content to HTML. +func GetAsciidocContent(content []byte) string { + cleanContent := bytes.Replace(content, SummaryDivider, []byte(""), 1) + + path, err := exec.LookPath("asciidoctor") + if err != nil { + path, err = exec.LookPath("asciidoc") + if err != nil { + jww.ERROR.Println("asciidoctor / asciidoc not found in $PATH: Please install.\n", + " Leaving AsciiDoc content unrendered.") + return (string(content)) + } + } + + jww.INFO.Println("Rendering with", path, "...") + cmd := exec.Command(path, "--safe", "-") + cmd.Stdin = bytes.NewReader(cleanContent) + var out bytes.Buffer + cmd.Stdout = &out + if err := cmd.Run(); err != nil { + jww.ERROR.Println(err) + } + + asciidocLines := strings.Split(out.String(), "\n") + for i, line := range asciidocLines { + if strings.HasPrefix(line, " 0 { + tmpContentWithTokensReplaced, err := replaceShortcodeTokens(tmpContent, shortcodePlaceholderPrefix, true, p.contentShortCodes) + + if err != nil { + jww.FATAL.Printf("Fail to replace short code tokens in %s:\n%s", p.BaseFileName(), err.Error()) + return HandledResult{err: err} + } else { + tmpContent = tmpContentWithTokensReplaced + } + } + + p.Content = helpers.BytesToHTML(tmpContent) + p.TableOfContents = helpers.BytesToHTML(tmpTableOfContents) //err := p.Convert() return HandledResult{page: p, err: nil} -- 2.30.2