Skip to content

Commit

Permalink
Docs
Browse files Browse the repository at this point in the history
  • Loading branch information
osteele committed Jun 29, 2017
1 parent fb26bb3 commit f15de87
Show file tree
Hide file tree
Showing 7 changed files with 58 additions and 30 deletions.
7 changes: 7 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,13 @@ cd expressions
ragel -Z scanner.rl && go test
```

Preview the documentation:

```bash
godoc -http=:6060&
open http://localhost:6060/pkg/github.com/osteele/liquid/
```

## References

<https://shopify.github.io/liquid>
Expand Down
4 changes: 4 additions & 0 deletions chunks/chunk.go
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
// Package chunks parses and evaluates template strings.
//
// A template string is composed of a sequence of chunk.
// Each chunk is either an object {{a.b}}, a tag {%if a>b%}, or a text chunk (anything outside of {{}} and {%%}.)
package chunks

import "fmt"
Expand Down
63 changes: 36 additions & 27 deletions engine.go
Original file line number Diff line number Diff line change
@@ -1,7 +1,5 @@
/*
Package liquid is a pure Go implementation of Shopify Liquid templates.
It's intended for use in for use in https://github.com/osteele/gojekyll.
Package liquid is a pure Go implementation of Shopify Liquid templates, for use in https://github.com/osteele/gojekyll.
See the project README https://github.com/osteele/liquid for additional information and implementation status.
*/
Expand All @@ -23,52 +21,63 @@ func init() {
}

// Engine parses template source into renderable text.
//
// An engine can be configured with additional filters and tags.
//
// Filters
//
// DefineFilter defines a Liquid filter.
//
// A filter is any function that takes at least one input, and returns one or two outputs.
// If it returns two outputs, the second must be an error.
type Engine interface {
// DefineFilter defines a filter function e.g. {{ value | filter: arg }}.
//
// Note: Although this function is defined on the engine, its effect is currently global.
DefineFilter(name string, fn interface{})
DefineTag(string, func(form string) (func(io.Writer, chunks.Context) error, error))

ParseTemplate(text []byte) (Template, error)
ParseAndRender(text []byte, bindings map[string]interface{}) ([]byte, error)
ParseAndRenderString(text string, bindings map[string]interface{}) (string, error)
// DefineTag defines a tag function e.g. {% tag %}.
//
// Note: Although this function is defined on the engine, its effect is currently global.
DefineTag(string, TagDefinition)

ParseTemplate(b []byte) (Template, error)
// ParseAndRender parses and then renders the template.
ParseAndRender(b []byte, bindings map[string]interface{}) ([]byte, error)
// ParseAndRenderString is a convenience wrapper for ParseAndRender, that has string input and output.
ParseAndRenderString(s string, bindings map[string]interface{}) (string, error)
}

type TagDefinition func(expr string) (func(io.Writer, chunks.Context) error, error)
// Renderer is the type of a function that is evaluated within a context and writes to output.
type Renderer func(io.Writer, chunks.Context) error

// TagDefinition is the type of a function that parses the argument string "args" from a tag "{% tagname args %}",
// and returns a renderer.
type TagDefinition func(parameters string) (func(io.Writer, chunks.Context) error, error)

type engine struct{}

type template struct {
ast chunks.ASTNode
}

// NewEngine returns a new engine.
// NewEngine returns a new template engine.
func NewEngine() Engine {
return engine{}
}

// DefineFilter defines a Liquid filter.
//
// A filter is any function that takes at least one input, and returns one or two outputs.
// If it returns two outputs, the second must be an error.
//
// Note: Although this function is defined on the engine, its effect is currently global.
// DefineFilter is in the Engine interface.
func (e engine) DefineFilter(name string, fn interface{}) {
// TODO define this on the engine, not globally
expressions.DefineFilter(name, fn)
}

// DefineTag defines a Liquid filter.
//
// A tag is any function that takes at least one input, and returns one or two outputs.
// If it returns two outputs, the second must be an error.
//
// Note: This interface is likely to change.
//
// Note: Although this function is defined on the engine, its effect is currently global.
func (e engine) DefineTag(name string, td func(form string) (func(io.Writer, chunks.Context) error, error)) {
// ParseAndRenderString is in the Engine interface.
func (e engine) DefineTag(name string, td TagDefinition) {
// TODO define this on the engine, not globally
chunks.DefineTag(name, chunks.TagDefinition(td))
}

// ParseTemplate is in the Engine interface.
func (e engine) ParseTemplate(text []byte) (Template, error) {
tokens := chunks.Scan(string(text), "")
ast, err := chunks.Parse(tokens)
Expand All @@ -78,7 +87,7 @@ func (e engine) ParseTemplate(text []byte) (Template, error) {
return &template{ast}, nil
}

// ParseAndRender parses and then renders the template.
// ParseAndRender is in the Engine interface.
func (e engine) ParseAndRender(text []byte, bindings map[string]interface{}) ([]byte, error) {
t, err := e.ParseTemplate(text)
if err != nil {
Expand All @@ -87,7 +96,7 @@ func (e engine) ParseAndRender(text []byte, bindings map[string]interface{}) ([]
return t.Render(bindings)
}

// ParseAndRenderString is a convenience wrapper for ParseAndRender, that has string input and output.
// ParseAndRenderString is in the Engine interface.
func (e engine) ParseAndRenderString(text string, bindings map[string]interface{}) (string, error) {
b, err := e.ParseAndRender([]byte(text), bindings)
if err != nil {
Expand Down
1 change: 1 addition & 0 deletions expressions/expressions.go
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
// Package expressions parses and evaluates the expression language that is used in objects {{a.b[c]}} and tag parameters {%assign pages = site.pages | reverse%}.
package expressions

import (
Expand Down
2 changes: 1 addition & 1 deletion expressions/scanner.rl
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
// Adapted from https://github.com/mhamrah/thermostat
package expressions
// Adapted from https://github.com/mhamrah/thermostat

import "fmt"
import "strconv"
Expand Down
7 changes: 7 additions & 0 deletions generics/generics.go
Original file line number Diff line number Diff line change
@@ -1,3 +1,10 @@
// Package generics defines methods such as sorting, comparison, and type conversion, that apply to interface types.
//
// It is similar to, and makes heavy use of, the reflect package.
//
// Since the intent is to provide runtime services for the Liquid expression interpreter,
// this package does not implement "generic" generics.
// It attempts to implement Liquid semantics (which are largely Ruby semantics).
package generics

import (
Expand Down
4 changes: 2 additions & 2 deletions liquid_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -40,12 +40,12 @@ func TestChunkParser(t *testing.T) {
func Example() {
engine := NewEngine()
template := `<h1>{{page.title}}</h1>`
scope := map[string]interface{}{
bindings := map[string]interface{}{
"page": map[string]interface{}{
"title": "Introduction",
},
}
out, err := engine.ParseAndRenderString(template, scope)
out, err := engine.ParseAndRenderString(template, bindings)
if err != nil {
log.Fatalln(err)
}
Expand Down

0 comments on commit f15de87

Please sign in to comment.