297 lines
9.9 KiB
YAML
297 lines
9.9 KiB
YAML
overview: |
|
||
Interpolation tags are used to integrate dynamic content into the template.
|
||
|
||
The tag's content MUST be a non-whitespace character sequence NOT containing
|
||
the current closing delimiter.
|
||
|
||
This tag's content names the data to replace the tag. A single period (`.`)
|
||
indicates that the item currently sitting atop the context stack should be
|
||
used; otherwise, name resolution is as follows:
|
||
1) Split the name on periods; the first part is the name to resolve, any
|
||
remaining parts should be retained.
|
||
2) Walk the context stack from top to bottom, finding the first context
|
||
that is a) a hash containing the name as a key OR b) an object responding
|
||
to a method with the given name.
|
||
3) If the context is a hash, the data is the value associated with the
|
||
name.
|
||
4) If the context is an object, the data is the value returned by the
|
||
method with the given name.
|
||
5) If any name parts were retained in step 1, each should be resolved
|
||
against a context stack containing only the result from the former
|
||
resolution. If any part fails resolution, the result should be considered
|
||
falsey, and should interpolate as the empty string.
|
||
Data should be coerced into a string (and escaped, if appropriate) before
|
||
interpolation.
|
||
|
||
The Interpolation tags MUST NOT be treated as standalone.
|
||
tests:
|
||
- name: No Interpolation
|
||
desc: Mustache-free templates should render as-is.
|
||
data: { }
|
||
template: |
|
||
Hello from {Mustache}!
|
||
expected: |
|
||
Hello from {Mustache}!
|
||
|
||
- name: Basic Interpolation
|
||
desc: Unadorned tags should interpolate content into the template.
|
||
data: { subject: "world" }
|
||
template: |
|
||
Hello, {{subject}}!
|
||
expected: |
|
||
Hello, world!
|
||
|
||
- name: HTML Escaping
|
||
desc: Basic interpolation should be HTML escaped.
|
||
data: { forbidden: '& " < >' }
|
||
template: |
|
||
These characters should be HTML escaped: {{forbidden}}
|
||
expected: |
|
||
These characters should be HTML escaped: & " < >
|
||
|
||
- name: Triple Mustache
|
||
desc: Triple mustaches should interpolate without HTML escaping.
|
||
data: { forbidden: '& " < >' }
|
||
template: |
|
||
These characters should not be HTML escaped: {{{forbidden}}}
|
||
expected: |
|
||
These characters should not be HTML escaped: & " < >
|
||
|
||
- name: Ampersand
|
||
desc: Ampersand should interpolate without HTML escaping.
|
||
data: { forbidden: '& " < >' }
|
||
template: |
|
||
These characters should not be HTML escaped: {{&forbidden}}
|
||
expected: |
|
||
These characters should not be HTML escaped: & " < >
|
||
|
||
- name: Basic Integer Interpolation
|
||
desc: Integers should interpolate seamlessly.
|
||
data: { mph: 85 }
|
||
template: '"{{mph}} miles an hour!"'
|
||
expected: '"85 miles an hour!"'
|
||
|
||
- name: Triple Mustache Integer Interpolation
|
||
desc: Integers should interpolate seamlessly.
|
||
data: { mph: 85 }
|
||
template: '"{{{mph}}} miles an hour!"'
|
||
expected: '"85 miles an hour!"'
|
||
|
||
- name: Ampersand Integer Interpolation
|
||
desc: Integers should interpolate seamlessly.
|
||
data: { mph: 85 }
|
||
template: '"{{&mph}} miles an hour!"'
|
||
expected: '"85 miles an hour!"'
|
||
|
||
- name: Basic Decimal Interpolation
|
||
desc: Decimals should interpolate seamlessly with proper significance.
|
||
data: { power: 1.210 }
|
||
template: '"{{power}} jiggawatts!"'
|
||
expected: '"1.21 jiggawatts!"'
|
||
|
||
- name: Triple Mustache Decimal Interpolation
|
||
desc: Decimals should interpolate seamlessly with proper significance.
|
||
data: { power: 1.210 }
|
||
template: '"{{{power}}} jiggawatts!"'
|
||
expected: '"1.21 jiggawatts!"'
|
||
|
||
- name: Ampersand Decimal Interpolation
|
||
desc: Decimals should interpolate seamlessly with proper significance.
|
||
data: { power: 1.210 }
|
||
template: '"{{&power}} jiggawatts!"'
|
||
expected: '"1.21 jiggawatts!"'
|
||
|
||
- name: Basic Null Interpolation
|
||
desc: Nulls should interpolate as the empty string.
|
||
data: { cannot: null }
|
||
template: "I ({{cannot}}) be seen!"
|
||
expected: "I () be seen!"
|
||
|
||
- name: Triple Mustache Null Interpolation
|
||
desc: Nulls should interpolate as the empty string.
|
||
data: { cannot: null }
|
||
template: "I ({{{cannot}}}) be seen!"
|
||
expected: "I () be seen!"
|
||
|
||
- name: Ampersand Null Interpolation
|
||
desc: Nulls should interpolate as the empty string.
|
||
data: { cannot: null }
|
||
template: "I ({{&cannot}}) be seen!"
|
||
expected: "I () be seen!"
|
||
|
||
# Context Misses
|
||
|
||
- name: Basic Context Miss Interpolation
|
||
desc: Failed context lookups should default to empty strings.
|
||
data: { }
|
||
template: "I ({{cannot}}) be seen!"
|
||
expected: "I () be seen!"
|
||
|
||
- name: Triple Mustache Context Miss Interpolation
|
||
desc: Failed context lookups should default to empty strings.
|
||
data: { }
|
||
template: "I ({{{cannot}}}) be seen!"
|
||
expected: "I () be seen!"
|
||
|
||
- name: Ampersand Context Miss Interpolation
|
||
desc: Failed context lookups should default to empty strings.
|
||
data: { }
|
||
template: "I ({{&cannot}}) be seen!"
|
||
expected: "I () be seen!"
|
||
|
||
# Dotted Names
|
||
|
||
- name: Dotted Names - Basic Interpolation
|
||
desc: Dotted names should be considered a form of shorthand for sections.
|
||
data: { person: { name: 'Joe' } }
|
||
template: '"{{person.name}}" == "{{#person}}{{name}}{{/person}}"'
|
||
expected: '"Joe" == "Joe"'
|
||
|
||
- name: Dotted Names - Triple Mustache Interpolation
|
||
desc: Dotted names should be considered a form of shorthand for sections.
|
||
data: { person: { name: 'Joe' } }
|
||
template: '"{{{person.name}}}" == "{{#person}}{{{name}}}{{/person}}"'
|
||
expected: '"Joe" == "Joe"'
|
||
|
||
- name: Dotted Names - Ampersand Interpolation
|
||
desc: Dotted names should be considered a form of shorthand for sections.
|
||
data: { person: { name: 'Joe' } }
|
||
template: '"{{&person.name}}" == "{{#person}}{{&name}}{{/person}}"'
|
||
expected: '"Joe" == "Joe"'
|
||
|
||
- name: Dotted Names - Arbitrary Depth
|
||
desc: Dotted names should be functional to any level of nesting.
|
||
data:
|
||
a: { b: { c: { d: { e: { name: 'Phil' } } } } }
|
||
template: '"{{a.b.c.d.e.name}}" == "Phil"'
|
||
expected: '"Phil" == "Phil"'
|
||
|
||
- name: Dotted Names - Broken Chains
|
||
desc: Any falsey value prior to the last part of the name should yield ''.
|
||
data:
|
||
a: { }
|
||
template: '"{{a.b.c}}" == ""'
|
||
expected: '"" == ""'
|
||
|
||
- name: Dotted Names - Broken Chain Resolution
|
||
desc: Each part of a dotted name should resolve only against its parent.
|
||
data:
|
||
a: { b: { } }
|
||
c: { name: 'Jim' }
|
||
template: '"{{a.b.c.name}}" == ""'
|
||
expected: '"" == ""'
|
||
|
||
- name: Dotted Names - Initial Resolution
|
||
desc: The first part of a dotted name should resolve as any other name.
|
||
data:
|
||
a: { b: { c: { d: { e: { name: 'Phil' } } } } }
|
||
b: { c: { d: { e: { name: 'Wrong' } } } }
|
||
template: '"{{#a}}{{b.c.d.e.name}}{{/a}}" == "Phil"'
|
||
expected: '"Phil" == "Phil"'
|
||
|
||
- name: Dotted Names - Context Precedence
|
||
desc: Dotted names should be resolved against former resolutions.
|
||
data:
|
||
a: { b: { } }
|
||
b: { c: 'ERROR' }
|
||
template: '{{#a}}{{b.c}}{{/a}}'
|
||
expected: ''
|
||
|
||
# Implicit Iterators
|
||
|
||
- name: Implicit Iterators - Basic Interpolation
|
||
desc: Unadorned tags should interpolate content into the template.
|
||
data: "world"
|
||
template: |
|
||
Hello, {{.}}!
|
||
expected: |
|
||
Hello, world!
|
||
|
||
- name: Implicit Iterators - HTML Escaping
|
||
desc: Basic interpolation should be HTML escaped.
|
||
data: '& " < >'
|
||
template: |
|
||
These characters should be HTML escaped: {{.}}
|
||
expected: |
|
||
These characters should be HTML escaped: & " < >
|
||
|
||
- name: Implicit Iterators - Triple Mustache
|
||
desc: Triple mustaches should interpolate without HTML escaping.
|
||
data: '& " < >'
|
||
template: |
|
||
These characters should not be HTML escaped: {{{.}}}
|
||
expected: |
|
||
These characters should not be HTML escaped: & " < >
|
||
|
||
- name: Implicit Iterators - Ampersand
|
||
desc: Ampersand should interpolate without HTML escaping.
|
||
data: '& " < >'
|
||
template: |
|
||
These characters should not be HTML escaped: {{&.}}
|
||
expected: |
|
||
These characters should not be HTML escaped: & " < >
|
||
|
||
- name: Implicit Iterators - Basic Integer Interpolation
|
||
desc: Integers should interpolate seamlessly.
|
||
data: 85
|
||
template: '"{{.}} miles an hour!"'
|
||
expected: '"85 miles an hour!"'
|
||
|
||
# Whitespace Sensitivity
|
||
|
||
- name: Interpolation - Surrounding Whitespace
|
||
desc: Interpolation should not alter surrounding whitespace.
|
||
data: { string: '---' }
|
||
template: '| {{string}} |'
|
||
expected: '| --- |'
|
||
|
||
- name: Triple Mustache - Surrounding Whitespace
|
||
desc: Interpolation should not alter surrounding whitespace.
|
||
data: { string: '---' }
|
||
template: '| {{{string}}} |'
|
||
expected: '| --- |'
|
||
|
||
- name: Ampersand - Surrounding Whitespace
|
||
desc: Interpolation should not alter surrounding whitespace.
|
||
data: { string: '---' }
|
||
template: '| {{&string}} |'
|
||
expected: '| --- |'
|
||
|
||
- name: Interpolation - Standalone
|
||
desc: Standalone interpolation should not alter surrounding whitespace.
|
||
data: { string: '---' }
|
||
template: " {{string}}\n"
|
||
expected: " ---\n"
|
||
|
||
- name: Triple Mustache - Standalone
|
||
desc: Standalone interpolation should not alter surrounding whitespace.
|
||
data: { string: '---' }
|
||
template: " {{{string}}}\n"
|
||
expected: " ---\n"
|
||
|
||
- name: Ampersand - Standalone
|
||
desc: Standalone interpolation should not alter surrounding whitespace.
|
||
data: { string: '---' }
|
||
template: " {{&string}}\n"
|
||
expected: " ---\n"
|
||
|
||
# Whitespace Insensitivity
|
||
|
||
- name: Interpolation With Padding
|
||
desc: Superfluous in-tag whitespace should be ignored.
|
||
data: { string: "---" }
|
||
template: '|{{ string }}|'
|
||
expected: '|---|'
|
||
|
||
- name: Triple Mustache With Padding
|
||
desc: Superfluous in-tag whitespace should be ignored.
|
||
data: { string: "---" }
|
||
template: '|{{{ string }}}|'
|
||
expected: '|---|'
|
||
|
||
- name: Ampersand With Padding
|
||
desc: Superfluous in-tag whitespace should be ignored.
|
||
data: { string: "---" }
|
||
template: '|{{& string }}|'
|
||
expected: '|---|'
|