mirror of
https://github.com/kemko/liquid.git
synced 2026-01-04 17:25:41 +03:00
Compare commits
9 Commits
liquid_c_e
...
cache_var_
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
717174a57a | ||
|
|
b8fbd2b4fa | ||
|
|
ba5a9f2e47 | ||
|
|
1e309ba74b | ||
|
|
485340713a | ||
|
|
2af4ea1295 | ||
|
|
c5dfcd29b0 | ||
|
|
f9c289372d | ||
|
|
0b36540b78 |
@@ -20,14 +20,15 @@
|
|||||||
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
|
WordRegex = RUBY_VERSION < "1.9" ? '\w' : '[[:word:]]'
|
||||||
FilterSeparator = /\|/
|
FilterSeparator = /\|/
|
||||||
ArgumentSeparator = ','
|
ArgumentSeparator = ','
|
||||||
FilterArgumentSeparator = ':'
|
FilterArgumentSeparator = ':'
|
||||||
VariableAttributeSeparator = '.'
|
VariableAttributeSeparator = '.'
|
||||||
TagStart = /\{\%/
|
TagStart = /\{\%/
|
||||||
TagEnd = /\%\}/
|
TagEnd = /\%\}/
|
||||||
VariableSignature = /\(?[\w\-\.\[\]]\)?/
|
VariableSignature = /\(?[#{WordRegex}\-\.\[\]]\)?/o
|
||||||
VariableSegment = /[\w\-]/
|
VariableSegment = /[#{WordRegex}\-]/o
|
||||||
VariableStart = /\{\{/
|
VariableStart = /\{\{/
|
||||||
VariableEnd = /\}\}/
|
VariableEnd = /\}\}/
|
||||||
VariableIncompleteEnd = /\}\}?/
|
VariableIncompleteEnd = /\}\}?/
|
||||||
@@ -38,7 +39,7 @@ module Liquid
|
|||||||
OtherFilterArgument = /#{ArgumentSeparator}(?:#{StrictQuotedFragment})/o
|
OtherFilterArgument = /#{ArgumentSeparator}(?:#{StrictQuotedFragment})/o
|
||||||
SpacelessFilter = /^(?:'[^']+'|"[^"]+"|[^'"])*#{FilterSeparator}(?:#{StrictQuotedFragment})(?:#{FirstFilterArgument}(?:#{OtherFilterArgument})*)?/o
|
SpacelessFilter = /^(?:'[^']+'|"[^"]+"|[^'"])*#{FilterSeparator}(?:#{StrictQuotedFragment})(?:#{FirstFilterArgument}(?:#{OtherFilterArgument})*)?/o
|
||||||
Expression = /(?:#{QuotedFragment}(?:#{SpacelessFilter})*)/o
|
Expression = /(?:#{QuotedFragment}(?:#{SpacelessFilter})*)/o
|
||||||
TagAttributes = /(\w+)\s*\:\s*(#{QuotedFragment})/o
|
TagAttributes = /(#{WordRegex}+)\s*\:\s*(#{QuotedFragment})/o
|
||||||
AnyStartingTag = /\{\{|\{\%/
|
AnyStartingTag = /\{\{|\{\%/
|
||||||
PartialTemplateParser = /#{TagStart}.*?#{TagEnd}|#{VariableStart}.*?#{VariableIncompleteEnd}/o
|
PartialTemplateParser = /#{TagStart}.*?#{TagEnd}|#{VariableStart}.*?#{VariableIncompleteEnd}/o
|
||||||
TemplateParser = /(#{PartialTemplateParser}|#{AnyStartingTag})/o
|
TemplateParser = /(#{PartialTemplateParser}|#{AnyStartingTag})/o
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ module Liquid
|
|||||||
class Block < Tag
|
class Block < Tag
|
||||||
IsTag = /^#{TagStart}/o
|
IsTag = /^#{TagStart}/o
|
||||||
IsVariable = /^#{VariableStart}/o
|
IsVariable = /^#{VariableStart}/o
|
||||||
FullToken = /^#{TagStart}\s*(\w+)\s*(.*)?#{TagEnd}$/o
|
FullToken = /^#{TagStart}\s*(#{WordRegex}+)\s*(.*)?#{TagEnd}$/o
|
||||||
ContentOfVariable = /^#{VariableStart}(.*)#{VariableEnd}$/o
|
ContentOfVariable = /^#{VariableStart}(.*)#{VariableEnd}$/o
|
||||||
|
|
||||||
def parse(tokens)
|
def parse(tokens)
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
module Liquid
|
module Liquid
|
||||||
|
|
||||||
|
|
||||||
# Context keeps the variable stack and resolves variables, as well as keywords
|
# Context keeps the variable stack and resolves variables, as well as keywords
|
||||||
#
|
#
|
||||||
# context['variable'] = 'testing'
|
# context['variable'] = 'testing'
|
||||||
@@ -24,6 +25,7 @@ module Liquid
|
|||||||
squash_instance_assigns_with_environments
|
squash_instance_assigns_with_environments
|
||||||
|
|
||||||
@interrupts = []
|
@interrupts = []
|
||||||
|
@variable_cache = {}
|
||||||
end
|
end
|
||||||
|
|
||||||
def strainer
|
def strainer
|
||||||
@@ -77,17 +79,20 @@ module Liquid
|
|||||||
|
|
||||||
# Push new local scope on the stack. use <tt>Context#stack</tt> instead
|
# Push new local scope on the stack. use <tt>Context#stack</tt> instead
|
||||||
def push(new_scope={})
|
def push(new_scope={})
|
||||||
|
@variable_cache = {}
|
||||||
@scopes.unshift(new_scope)
|
@scopes.unshift(new_scope)
|
||||||
raise StackLevelError, "Nesting too deep" if @scopes.length > 100
|
raise StackLevelError, "Nesting too deep" if @scopes.length > 100
|
||||||
end
|
end
|
||||||
|
|
||||||
# Merge a hash of variables in the current local scope
|
# Merge a hash of variables in the current local scope
|
||||||
def merge(new_scopes)
|
def merge(new_scopes)
|
||||||
|
@variable_cache = {}
|
||||||
@scopes[0].merge!(new_scopes)
|
@scopes[0].merge!(new_scopes)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Pop from the stack. use <tt>Context#stack</tt> instead
|
# Pop from the stack. use <tt>Context#stack</tt> instead
|
||||||
def pop
|
def pop
|
||||||
|
@variable_cache = {}
|
||||||
raise ContextError if @scopes.size == 1
|
raise ContextError if @scopes.size == 1
|
||||||
@scopes.shift
|
@scopes.shift
|
||||||
end
|
end
|
||||||
@@ -101,69 +106,112 @@ module Liquid
|
|||||||
#
|
#
|
||||||
# context['var] #=> nil
|
# context['var] #=> nil
|
||||||
def stack(new_scope={})
|
def stack(new_scope={})
|
||||||
|
@variable_cache = {}
|
||||||
push(new_scope)
|
push(new_scope)
|
||||||
yield
|
yield
|
||||||
ensure
|
ensure
|
||||||
pop
|
pop
|
||||||
|
@variable_cache = {}
|
||||||
end
|
end
|
||||||
|
|
||||||
def clear_instance_assigns
|
def clear_instance_assigns
|
||||||
|
@variable_cache = {}
|
||||||
@scopes[0] = {}
|
@scopes[0] = {}
|
||||||
end
|
end
|
||||||
|
|
||||||
# Only allow String, Numeric, Hash, Array, Proc, Boolean or <tt>Liquid::Drop</tt>
|
|
||||||
def []=(key, value)
|
# Look up variable, either resolve directly after considering the name. We can directly handle
|
||||||
@scopes[0][key] = value
|
# Strings, digits, floats and booleans (true,false).
|
||||||
|
# If no match is made we lookup the variable in the current scope and
|
||||||
|
# later move up to the parent blocks to see if we can resolve the variable somewhere up the tree.
|
||||||
|
# Some special keywords return symbols. Those symbols are to be called on the rhs object in expressions
|
||||||
|
#
|
||||||
|
# Example:
|
||||||
|
# products == empty #=> products.empty?
|
||||||
|
def resolve(key)
|
||||||
|
case key
|
||||||
|
when nil, ""
|
||||||
|
return nil
|
||||||
|
when "blank"
|
||||||
|
return :blank?
|
||||||
|
when "empty"
|
||||||
|
return :empty?
|
||||||
|
end
|
||||||
|
|
||||||
|
result = Parser.parse(key)
|
||||||
|
stack = []
|
||||||
|
|
||||||
|
result.each do |(sym, value)|
|
||||||
|
|
||||||
|
case sym
|
||||||
|
when :id
|
||||||
|
stack.push value
|
||||||
|
when :lookup
|
||||||
|
left = stack.pop
|
||||||
|
value = find_variable(left)
|
||||||
|
|
||||||
|
stack.push(harden(value))
|
||||||
|
when :range
|
||||||
|
right = stack.pop.to_i
|
||||||
|
left = stack.pop.to_i
|
||||||
|
|
||||||
|
stack.push (left..right)
|
||||||
|
when :buildin
|
||||||
|
left = stack.pop
|
||||||
|
value = invoke_buildin(left, value)
|
||||||
|
|
||||||
|
stack.push(harden(value))
|
||||||
|
when :call
|
||||||
|
left = stack.pop
|
||||||
|
right = stack.pop
|
||||||
|
value = lookup_and_evaluate(right, left)
|
||||||
|
|
||||||
|
stack.push(harden(value))
|
||||||
|
else
|
||||||
|
raise "unknown #{sym}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
return stack.first
|
||||||
end
|
end
|
||||||
|
|
||||||
def [](key)
|
|
||||||
resolve(key)
|
# Only allow String, Numeric, Hash, Array, Proc, Boolean or <tt>Liquid::Drop</tt>
|
||||||
|
def []=(key, value)
|
||||||
|
@variable_cache[key] = value
|
||||||
|
@scopes[0][key] = value
|
||||||
end
|
end
|
||||||
|
|
||||||
def has_key?(key)
|
def has_key?(key)
|
||||||
resolve(key) != nil
|
resolve(key) != nil
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
alias_method :[], :resolve
|
||||||
LITERALS = {
|
|
||||||
nil => nil, 'nil' => nil, 'null' => nil, '' => nil,
|
|
||||||
'true' => true,
|
|
||||||
'false' => false,
|
|
||||||
'blank' => :blank?,
|
|
||||||
'empty' => :empty?
|
|
||||||
}
|
|
||||||
|
|
||||||
# Look up variable, either resolve directly after considering the name. We can directly handle
|
private
|
||||||
# Strings, digits, floats and booleans (true,false).
|
|
||||||
# If no match is made we lookup the variable in the current scope and
|
def invoke_buildin(obj, key)
|
||||||
# later move up to the parent blocks to see if we can resolve the variable somewhere up the tree.
|
# as weird as this is, liquid unit tests demand that we prioritize hash lookups
|
||||||
# Some special keywords return symbols. Those symbols are to be called on the rhs object in expressions
|
# to buildins. So if we got a hash and it has a :first element we need to call that
|
||||||
#
|
# instead of sending the first message...
|
||||||
# Example:
|
|
||||||
# products == empty #=> products.empty?
|
if obj.respond_to?(:has_key?) && obj.has_key?(key)
|
||||||
def resolve(key)
|
return lookup_and_evaluate(obj, key)
|
||||||
if LITERALS.key?(key)
|
end
|
||||||
LITERALS[key]
|
|
||||||
|
if obj.respond_to?(key)
|
||||||
|
return obj.send(key)
|
||||||
else
|
else
|
||||||
case key
|
return nil
|
||||||
when /^'(.*)'$/ # Single quoted strings
|
|
||||||
$1
|
|
||||||
when /^"(.*)"$/ # Double quoted strings
|
|
||||||
$1
|
|
||||||
when /^(-?\d+)$/ # Integer and floats
|
|
||||||
$1.to_i
|
|
||||||
when /^\((\S+)\.\.(\S+)\)$/ # Ranges
|
|
||||||
(resolve($1).to_i..resolve($2).to_i)
|
|
||||||
when /^(-?\d[\d\.]+)$/ # Floats
|
|
||||||
$1.to_f
|
|
||||||
else
|
|
||||||
variable(key)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Fetches an object starting at the local scope and then moving up the hierachy
|
# Fetches an object starting at the local scope and then moving up the hierachy
|
||||||
def find_variable(key)
|
def find_variable(key)
|
||||||
|
if val = @variable_cache[key]
|
||||||
|
return val
|
||||||
|
end
|
||||||
|
|
||||||
scope = @scopes.find { |s| s.has_key?(key) }
|
scope = @scopes.find { |s| s.has_key?(key) }
|
||||||
|
|
||||||
if scope.nil?
|
if scope.nil?
|
||||||
@@ -177,72 +225,36 @@ module Liquid
|
|||||||
|
|
||||||
scope ||= @environments.last || @scopes.last
|
scope ||= @environments.last || @scopes.last
|
||||||
variable ||= lookup_and_evaluate(scope, key)
|
variable ||= lookup_and_evaluate(scope, key)
|
||||||
|
@variable_cache[key] = variable
|
||||||
variable = variable.to_liquid
|
|
||||||
variable.context = self if variable.respond_to?(:context=)
|
|
||||||
|
|
||||||
return variable
|
return variable
|
||||||
end
|
end
|
||||||
|
|
||||||
# Resolves namespaced queries gracefully.
|
|
||||||
#
|
|
||||||
# Example
|
|
||||||
# @context['hash'] = {"name" => 'tobi'}
|
|
||||||
# assert_equal 'tobi', @context['hash.name']
|
|
||||||
# assert_equal 'tobi', @context['hash["name"]']
|
|
||||||
def variable(markup)
|
|
||||||
parts = markup.scan(VariableParser)
|
|
||||||
square_bracketed = /^\[(.*)\]$/
|
|
||||||
|
|
||||||
first_part = parts.shift
|
|
||||||
|
|
||||||
if first_part =~ square_bracketed
|
|
||||||
first_part = resolve($1)
|
|
||||||
end
|
|
||||||
|
|
||||||
if object = find_variable(first_part)
|
|
||||||
|
|
||||||
parts.each do |part|
|
|
||||||
part = resolve($1) if part_resolved = (part =~ square_bracketed)
|
|
||||||
|
|
||||||
# If object is a hash- or array-like object we look for the
|
|
||||||
# presence of the key and if its available we return it
|
|
||||||
if object.respond_to?(:[]) and
|
|
||||||
((object.respond_to?(:has_key?) and object.has_key?(part)) or
|
|
||||||
(object.respond_to?(:fetch) and part.is_a?(Integer)))
|
|
||||||
|
|
||||||
# if its a proc we will replace the entry with the proc
|
|
||||||
res = lookup_and_evaluate(object, part)
|
|
||||||
object = res.to_liquid
|
|
||||||
|
|
||||||
# Some special cases. If the part wasn't in square brackets and
|
|
||||||
# no key with the same name was found we interpret following calls
|
|
||||||
# as commands and call them on the current object
|
|
||||||
elsif !part_resolved and object.respond_to?(part) and ['size', 'first', 'last'].include?(part)
|
|
||||||
|
|
||||||
object = object.send(part.intern).to_liquid
|
|
||||||
|
|
||||||
# No key was present with the desired value and it wasn't one of the directly supported
|
|
||||||
# keywords either. The only thing we got left is to return nil
|
|
||||||
else
|
|
||||||
return nil
|
|
||||||
end
|
|
||||||
|
|
||||||
# If we are dealing with a drop here we have to
|
|
||||||
object.context = self if object.respond_to?(:context=)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
object
|
|
||||||
end # variable
|
|
||||||
|
|
||||||
def lookup_and_evaluate(obj, key)
|
def lookup_and_evaluate(obj, key)
|
||||||
if (value = obj[key]).is_a?(Proc) && obj.respond_to?(:[]=)
|
return nil unless obj.respond_to?(:[])
|
||||||
obj[key] = (value.arity == 0) ? value.call : value.call(self)
|
|
||||||
else
|
if obj.is_a?(Array)
|
||||||
value
|
return nil unless key.is_a?(Integer)
|
||||||
end
|
end
|
||||||
end # lookup_and_evaluate
|
|
||||||
|
value = obj[key]
|
||||||
|
|
||||||
|
if value.is_a?(Proc)
|
||||||
|
# call the proc
|
||||||
|
value = (value.arity == 0) ? value.call : value.call(self)
|
||||||
|
|
||||||
|
# memozie if possible
|
||||||
|
obj[key] = value if obj.respond_to?(:[]=)
|
||||||
|
end
|
||||||
|
|
||||||
|
value
|
||||||
|
end
|
||||||
|
|
||||||
|
def harden(value)
|
||||||
|
value = value.to_liquid
|
||||||
|
value.context = self if value.respond_to?(:context=)
|
||||||
|
return value
|
||||||
|
end
|
||||||
|
|
||||||
def squash_instance_assigns_with_environments
|
def squash_instance_assigns_with_environments
|
||||||
@scopes.last.each_key do |k|
|
@scopes.last.each_key do |k|
|
||||||
@@ -254,6 +266,7 @@ module Liquid
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
end # squash_instance_assigns_with_environments
|
end # squash_instance_assigns_with_environments
|
||||||
|
|
||||||
end # Context
|
end # Context
|
||||||
|
|
||||||
end # Liquid
|
end # Liquid
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module Liquid
|
module Liquid
|
||||||
class TableRow < Block
|
class TableRow < Block
|
||||||
Syntax = /(\w+)\s+in\s+(#{QuotedFragment}+)/o
|
Syntax = /(#{WordRegex}+)\s+in\s+(#{QuotedFragment}+)/o
|
||||||
|
|
||||||
def initialize(tag_name, markup, tokens)
|
def initialize(tag_name, markup, tokens)
|
||||||
if markup =~ Syntax
|
if markup =~ Syntax
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ module Liquid
|
|||||||
# in a sidebar or footer.
|
# in a sidebar or footer.
|
||||||
#
|
#
|
||||||
class Capture < Block
|
class Capture < Block
|
||||||
Syntax = /(\w+)/
|
Syntax = /(#{WordRegex}+)/o
|
||||||
|
|
||||||
def initialize(tag_name, markup, tokens)
|
def initialize(tag_name, markup, tokens)
|
||||||
if markup =~ Syntax
|
if markup =~ Syntax
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ module Liquid
|
|||||||
# forloop.last:: Returns true if the item is the last item.
|
# forloop.last:: Returns true if the item is the last item.
|
||||||
#
|
#
|
||||||
class For < Block
|
class For < Block
|
||||||
Syntax = /(\w+)\s+in\s+(#{QuotedFragment}+)\s*(reversed)?/o
|
Syntax = /(#{WordRegex}+)\s+in\s+(#{QuotedFragment}+)\s*(reversed)?/ou
|
||||||
|
|
||||||
def initialize(tag_name, markup, tokens)
|
def initialize(tag_name, markup, tokens)
|
||||||
if markup =~ Syntax
|
if markup =~ Syntax
|
||||||
|
|||||||
@@ -2,10 +2,10 @@ module Liquid
|
|||||||
class Include < Tag
|
class Include < Tag
|
||||||
Syntax = /(#{QuotedFragment}+)(\s+(?:with|for)\s+(#{QuotedFragment}+))?/o
|
Syntax = /(#{QuotedFragment}+)(\s+(?:with|for)\s+(#{QuotedFragment}+))?/o
|
||||||
|
|
||||||
def initialize(tag_name, markup, tokens)
|
def initialize(tag_name, markup, tokens)
|
||||||
if markup =~ Syntax
|
if markup =~ Syntax
|
||||||
|
|
||||||
@template_name = $1
|
@template_name = $1
|
||||||
@variable_name = $3
|
@variable_name = $3
|
||||||
@attributes = {}
|
@attributes = {}
|
||||||
|
|
||||||
@@ -24,8 +24,7 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def render(context)
|
def render(context)
|
||||||
source = _read_template_from_file_system(context)
|
partial = load_cached_partial(context)
|
||||||
partial = Liquid::Template.parse(source)
|
|
||||||
variable = context[@variable_name || @template_name[1..-2]]
|
variable = context[@variable_name || @template_name[1..-2]]
|
||||||
|
|
||||||
context.stack do
|
context.stack do
|
||||||
@@ -46,7 +45,21 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def _read_template_from_file_system(context)
|
def load_cached_partial(context)
|
||||||
|
cached_partials = context.registers[:cached_partials] || {}
|
||||||
|
template_name = context[@template_name]
|
||||||
|
|
||||||
|
if cached = cached_partials[template_name]
|
||||||
|
return cached
|
||||||
|
end
|
||||||
|
source = read_template_from_file_system(context)
|
||||||
|
partial = Liquid::Template.parse(source)
|
||||||
|
cached_partials[template_name] = partial
|
||||||
|
context.registers[:cached_partials] = cached_partials
|
||||||
|
partial
|
||||||
|
end
|
||||||
|
|
||||||
|
def read_template_from_file_system(context)
|
||||||
file_system = context.registers[:file_system] || Liquid::Template.file_system
|
file_system = context.registers[:file_system] || Liquid::Template.file_system
|
||||||
|
|
||||||
# make read_template_file call backwards-compatible.
|
# make read_template_file call backwards-compatible.
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ module Liquid
|
|||||||
if match[2].match(/#{FilterSeparator}\s*(.*)/o)
|
if match[2].match(/#{FilterSeparator}\s*(.*)/o)
|
||||||
filters = Regexp.last_match(1).scan(FilterParser)
|
filters = Regexp.last_match(1).scan(FilterParser)
|
||||||
filters.each do |f|
|
filters.each do |f|
|
||||||
if matches = f.match(/\s*(\w+)(?:\s*#{FilterArgumentSeparator}(.*))?/)
|
if matches = f.match(/\s*(#{WordRegex}+)(?:\s*#{FilterArgumentSeparator}(.*))?/)
|
||||||
filtername = matches[1]
|
filtername = matches[1]
|
||||||
filterargs = matches[2].to_s.scan(/(?:\A|#{ArgumentSeparator})\s*((?:\w+\s*\:\s*)?#{QuotedFragment})/o).flatten
|
filterargs = matches[2].to_s.scan(/(?:\A|#{ArgumentSeparator})\s*((?:\w+\s*\:\s*)?#{QuotedFragment})/o).flatten
|
||||||
@filters << [filtername, filterargs]
|
@filters << [filtername, filterargs]
|
||||||
|
|||||||
@@ -14,6 +14,17 @@ require File.dirname(__FILE__) + '/shopify/liquid'
|
|||||||
require File.dirname(__FILE__) + '/shopify/database.rb'
|
require File.dirname(__FILE__) + '/shopify/database.rb'
|
||||||
|
|
||||||
class ThemeRunner
|
class ThemeRunner
|
||||||
|
class FileSystem
|
||||||
|
|
||||||
|
def initialize(path)
|
||||||
|
@path = path
|
||||||
|
end
|
||||||
|
|
||||||
|
# Called by Liquid to retrieve a template file
|
||||||
|
def read_template_file(template_path, context)
|
||||||
|
File.read(@path + '/' + template_path + '.liquid')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
# Load all templates into memory, do this now so that
|
# Load all templates into memory, do this now so that
|
||||||
# we don't profile IO.
|
# we don't profile IO.
|
||||||
@@ -47,7 +58,7 @@ class ThemeRunner
|
|||||||
|
|
||||||
# Compute page_tempalte outside of profiler run, uninteresting to profiler
|
# Compute page_tempalte outside of profiler run, uninteresting to profiler
|
||||||
page_template = File.basename(template_name, File.extname(template_name))
|
page_template = File.basename(template_name, File.extname(template_name))
|
||||||
compile_and_render(liquid, layout, assigns, page_template)
|
compile_and_render(liquid, layout, assigns, page_template, template_name)
|
||||||
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -74,7 +85,7 @@ class ThemeRunner
|
|||||||
html = nil
|
html = nil
|
||||||
|
|
||||||
RubyProf.resume
|
RubyProf.resume
|
||||||
html = compile_and_render(liquid, layout, assigns, page_template)
|
html = compile_and_render(liquid, layout, assigns, page_template, template_name)
|
||||||
RubyProf.pause
|
RubyProf.pause
|
||||||
|
|
||||||
|
|
||||||
@@ -88,10 +99,11 @@ class ThemeRunner
|
|||||||
RubyProf.stop
|
RubyProf.stop
|
||||||
end
|
end
|
||||||
|
|
||||||
def compile_and_render(template, layout, assigns, page_template)
|
def compile_and_render(template, layout, assigns, page_template, template_file)
|
||||||
tmpl = Liquid::Template.new
|
tmpl = Liquid::Template.new
|
||||||
tmpl.assigns['page_title'] = 'Page title'
|
tmpl.assigns['page_title'] = 'Page title'
|
||||||
tmpl.assigns['template'] = page_template
|
tmpl.assigns['template'] = page_template
|
||||||
|
tmpl.registers[:file_system] = ThemeRunner::FileSystem.new(File.dirname(template_file))
|
||||||
|
|
||||||
content_for_layout = tmpl.parse(template).render(assigns)
|
content_for_layout = tmpl.parse(template).render(assigns)
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,13 @@ class AssignTest < Test::Unit::TestCase
|
|||||||
'{% assign foo = values %}.{{ foo[1] }}.',
|
'{% assign foo = values %}.{{ foo[1] }}.',
|
||||||
'values' => %w{foo bar baz})
|
'values' => %w{foo bar baz})
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_assigned_utf8_variable
|
||||||
|
assert_template_result('.bar.',
|
||||||
|
"{% assign foo\u6000 = values %}.{{ foo\u6000[1] }}.",
|
||||||
|
'values' => %w{foo bar baz})
|
||||||
|
end
|
||||||
|
|
||||||
def test_assign_with_filter
|
def test_assign_with_filter
|
||||||
assert_template_result('.bar.',
|
assert_template_result('.bar.',
|
||||||
'{% assign foo = values | split: "," %}.{{ foo[1] }}.',
|
'{% assign foo = values | split: "," %}.{{ foo[1] }}.',
|
||||||
|
|||||||
@@ -51,6 +51,14 @@ class BlockTest < Test::Unit::TestCase
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_with_custom_utf8_tag
|
||||||
|
Liquid::Template.register_tag("testtag\u6000", Block)
|
||||||
|
|
||||||
|
assert_nothing_thrown do
|
||||||
|
template = Liquid::Template.parse( "{% testtag\u6000 something\u6000 %} {% endtesttag\u6000 %}")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def block_types(nodelist)
|
def block_types(nodelist)
|
||||||
nodelist.collect { |node| node.class }
|
nodelist.collect { |node| node.class }
|
||||||
|
|||||||
@@ -7,6 +7,10 @@ class CaptureTest < Test::Unit::TestCase
|
|||||||
assert_template_result("test string", "{% capture 'var' %}test string{% endcapture %}{{var}}", {})
|
assert_template_result("test string", "{% capture 'var' %}test string{% endcapture %}{{var}}", {})
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_captures_block_content_in_utf8_variable
|
||||||
|
assert_template_result("test string", "{% capture var\u6000 %}test string{% endcapture %}{{var\u6000}}", {})
|
||||||
|
end
|
||||||
|
|
||||||
def test_capture_to_variable_from_outer_scope_if_existing
|
def test_capture_to_variable_from_outer_scope_if_existing
|
||||||
template_source = <<-END_TEMPLATE
|
template_source = <<-END_TEMPLATE
|
||||||
{% assign var = '' %}
|
{% assign var = '' %}
|
||||||
|
|||||||
@@ -98,6 +98,11 @@ class ContextTest < Test::Unit::TestCase
|
|||||||
assert_equal nil, @context['nil']
|
assert_equal nil, @context['nil']
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_utf8_variables
|
||||||
|
@context["chinese\u6000variable"] = 'chinese'
|
||||||
|
assert_equal 'chinese', @context["chinese\u6000variable"]
|
||||||
|
end
|
||||||
|
|
||||||
def test_variables_not_existing
|
def test_variables_not_existing
|
||||||
assert_equal nil, @context['does_not_exist']
|
assert_equal nil, @context['does_not_exist']
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -25,6 +25,11 @@ HERE
|
|||||||
assert_template_result(expected,template,'array' => [1,2,3])
|
assert_template_result(expected,template,'array' => [1,2,3])
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_utf8_for
|
||||||
|
assigns = {"array\u6000chinese" => [1,2,3]}
|
||||||
|
assert_template_result('123', "{% for item\u6000chinese in array\u6000chinese %}{{ item\u6000chinese }}{% endfor %}", assigns)
|
||||||
|
end
|
||||||
|
|
||||||
def test_for_reversed
|
def test_for_reversed
|
||||||
assigns = {'array' => [ 1, 2, 3] }
|
assigns = {'array' => [ 1, 2, 3] }
|
||||||
assert_template_result('321','{%for item in array reversed %}{{item}}{%endfor%}',assigns)
|
assert_template_result('321','{%for item in array reversed %}{{item}}{%endfor%}',assigns)
|
||||||
|
|||||||
@@ -26,6 +26,12 @@ class HtmlTagTest < Test::Unit::TestCase
|
|||||||
'numbers' => [])
|
'numbers' => [])
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_utf8_html_table
|
||||||
|
assert_template_result("<tr class=\"row1\">\n<td class=\"col1\"> 1 </td></tr>\n",
|
||||||
|
"{% tablerow n\u6000 in numbers\u6000 %} {{n\u6000}} {% endtablerow %}",
|
||||||
|
"numbers\u6000" => [1])
|
||||||
|
end
|
||||||
|
|
||||||
def test_html_table_with_different_cols
|
def test_html_table_with_different_cols
|
||||||
assert_template_result("<tr class=\"row1\">\n<td class=\"col1\"> 1 </td><td class=\"col2\"> 2 </td><td class=\"col3\"> 3 </td><td class=\"col4\"> 4 </td><td class=\"col5\"> 5 </td></tr>\n<tr class=\"row2\"><td class=\"col1\"> 6 </td></tr>\n",
|
assert_template_result("<tr class=\"row1\">\n<td class=\"col1\"> 1 </td><td class=\"col2\"> 2 </td><td class=\"col3\"> 3 </td><td class=\"col4\"> 4 </td><td class=\"col5\"> 5 </td></tr>\n<tr class=\"row2\"><td class=\"col1\"> 6 </td></tr>\n",
|
||||||
'{% tablerow n in numbers cols:5%} {{n}} {% endtablerow %}',
|
'{% tablerow n in numbers cols:5%} {{n}} {% endtablerow %}',
|
||||||
|
|||||||
@@ -39,6 +39,15 @@ class OtherFileSystem
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class CountingFileSystem
|
||||||
|
attr_reader :count
|
||||||
|
def read_template_file(template_path, context)
|
||||||
|
@count ||= 0
|
||||||
|
@count += 1
|
||||||
|
'from CountingFileSystem'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
class IncludeTagTest < Test::Unit::TestCase
|
class IncludeTagTest < Test::Unit::TestCase
|
||||||
include Liquid
|
include Liquid
|
||||||
|
|
||||||
@@ -136,4 +145,22 @@ class IncludeTagTest < Test::Unit::TestCase
|
|||||||
|
|
||||||
assert_equal "Product: Draft 151cm ", Template.parse("{% include template for product %}").render("template" => 'product', 'product' => { 'title' => 'Draft 151cm'})
|
assert_equal "Product: Draft 151cm ", Template.parse("{% include template for product %}").render("template" => 'product', 'product' => { 'title' => 'Draft 151cm'})
|
||||||
end
|
end
|
||||||
end # IncludeTagTest
|
|
||||||
|
def test_include_tag_caches_second_read_of_same_partial
|
||||||
|
file_system = CountingFileSystem.new
|
||||||
|
assert_equal 'from CountingFileSystemfrom CountingFileSystem',
|
||||||
|
Template.parse("{% include 'pick_a_source' %}{% include 'pick_a_source' %}").render({}, :registers => {:file_system => file_system})
|
||||||
|
assert_equal 1, file_system.count
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_include_tag_doesnt_cache_partials_across_renders
|
||||||
|
file_system = CountingFileSystem.new
|
||||||
|
assert_equal 'from CountingFileSystem',
|
||||||
|
Template.parse("{% include 'pick_a_source' %}").render({}, :registers => {:file_system => file_system})
|
||||||
|
assert_equal 1, file_system.count
|
||||||
|
|
||||||
|
assert_equal 'from CountingFileSystem',
|
||||||
|
Template.parse("{% include 'pick_a_source' %}").render({}, :registers => {:file_system => file_system})
|
||||||
|
assert_equal 2, file_system.count
|
||||||
|
end
|
||||||
|
end # IncludeTagTest
|
||||||
|
|||||||
@@ -50,6 +50,12 @@ class VariableTest < Test::Unit::TestCase
|
|||||||
assert_equal [["things",["\"%Y, okay?\"","'the other one'"]]], var.filters
|
assert_equal [["things",["\"%Y, okay?\"","'the other one'"]]], var.filters
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_utf8_filters
|
||||||
|
var = Variable.new("foo | chinese\u6000filter: value\u6000")
|
||||||
|
assert_equal 'foo', var.name
|
||||||
|
assert_equal [["chinese\u6000filter",["value\u6000"]]], var.filters
|
||||||
|
end
|
||||||
|
|
||||||
def test_filter_with_date_parameter
|
def test_filter_with_date_parameter
|
||||||
|
|
||||||
var = Variable.new(%! '2006-06-06' | date: "%m/%d/%Y"!)
|
var = Variable.new(%! '2006-06-06' | date: "%m/%d/%Y"!)
|
||||||
|
|||||||
Reference in New Issue
Block a user