mirror of
https://github.com/kemko/liquid.git
synced 2026-01-02 16:25:42 +03:00
Compare commits
185 Commits
sort-numer
...
inline-com
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0fc45ca3af | ||
|
|
ab455480fa | ||
|
|
5e8e5e89b1 | ||
|
|
6ddfaec3f9 | ||
|
|
fff6c565c1 | ||
|
|
31f7be8a6d | ||
|
|
dcb5a67089 | ||
|
|
efe44a7e6a | ||
|
|
8625e66453 | ||
|
|
3cae09b968 | ||
|
|
3c499d0241 | ||
|
|
e71e53ffb5 | ||
|
|
260c863e23 | ||
|
|
42b6c07cd0 | ||
|
|
c91a6827f2 | ||
|
|
5dbc3d5701 | ||
|
|
22683cbd2a | ||
|
|
abfab3bef2 | ||
|
|
51e8d6234a | ||
|
|
7ca2846d9c | ||
|
|
7ba0fc7952 | ||
|
|
e2c86d137f | ||
|
|
776a63b61d | ||
|
|
84f9d6957c | ||
|
|
7d32728e16 | ||
|
|
40a9b72b3c | ||
|
|
4ff26cd707 | ||
|
|
462919a28f | ||
|
|
f3e2be9f85 | ||
|
|
4d40f83457 | ||
|
|
00be1e4dd4 | ||
|
|
f7d67b946e | ||
|
|
ae9aee896b | ||
|
|
6dec172743 | ||
|
|
da581d988a | ||
|
|
7960826552 | ||
|
|
84059691b8 | ||
|
|
896288eff1 | ||
|
|
b3f132efd1 | ||
|
|
60214b957c | ||
|
|
7361220af6 | ||
|
|
cb2ad71a31 | ||
|
|
900e3a6491 | ||
|
|
f18084203d | ||
|
|
3358a892f2 | ||
|
|
bbfcaa2cc0 | ||
|
|
ba657871bc | ||
|
|
29d5d9674a | ||
|
|
0a645e72c1 | ||
|
|
1850511334 | ||
|
|
300adfd7ae | ||
|
|
f357662f37 | ||
|
|
ed0aebcbc9 | ||
|
|
ea4f1885f8 | ||
|
|
2f75db604f | ||
|
|
d844a3dd8b | ||
|
|
9fcba1a26c | ||
|
|
0659891e68 | ||
|
|
e7fb3b18f3 | ||
|
|
e6eef4b2c4 | ||
|
|
2ce577e36b | ||
|
|
c7c21e88f0 | ||
|
|
a89371b0b9 | ||
|
|
8f7f8761d1 | ||
|
|
a3ff300419 | ||
|
|
ea6e326b9c | ||
|
|
740f8759cc | ||
|
|
bb9cd4eb6a | ||
|
|
3a591fbf26 | ||
|
|
7754d5aef5 | ||
|
|
1d63d5db5f | ||
|
|
26640368e5 | ||
|
|
f23c2a83f2 | ||
|
|
61d54d1b19 | ||
|
|
10ea6144e0 | ||
|
|
292d971937 | ||
|
|
5c082472a1 | ||
|
|
0bedc71854 | ||
|
|
fe66edb825 | ||
|
|
bfa2df7036 | ||
|
|
0e52706a5b | ||
|
|
4c6166f989 | ||
|
|
8e99b3bd7f | ||
|
|
f6532de1fd | ||
|
|
001fde7694 | ||
|
|
b872eac2b9 | ||
|
|
038d0585cf | ||
|
|
b15428ea83 | ||
|
|
c9ad9d338c | ||
|
|
ae6bd9f6b0 | ||
|
|
866e437c05 | ||
|
|
784db053f2 | ||
|
|
ff1c6bd26e | ||
|
|
46fd63da5f | ||
|
|
420a1c79e1 | ||
|
|
6d39050e1e | ||
|
|
077bf2a409 | ||
|
|
1a3e38c018 | ||
|
|
e495f75cc2 | ||
|
|
e781449c36 | ||
|
|
7eb03ea198 | ||
|
|
bd34cd5613 | ||
|
|
c28d455f7b | ||
|
|
d250a7f502 | ||
|
|
b0f46326ca | ||
|
|
7aed2f122c | ||
|
|
5199a34d9b | ||
|
|
4c2ab6f878 | ||
|
|
a818dd9d19 | ||
|
|
efef03d944 | ||
|
|
33760f083a | ||
|
|
013802c877 | ||
|
|
3dcad3b3cd | ||
|
|
db065315ba | ||
|
|
a03f02789b | ||
|
|
ca4b9b43af | ||
|
|
77084930e9 | ||
|
|
fb77921b15 | ||
|
|
0d02dea20b | ||
|
|
86b47ba28b | ||
|
|
95ff0595c6 | ||
|
|
bbc56f35ec | ||
|
|
dfbbf87ba9 | ||
|
|
037b603603 | ||
|
|
bd33df09de | ||
|
|
6ca5b62112 | ||
|
|
e1a2057a1b | ||
|
|
ae9dbe0ca7 | ||
|
|
3b486425b0 | ||
|
|
b08bcf00ac | ||
|
|
0740e8b431 | ||
|
|
5532df880f | ||
|
|
2b11efc3ae | ||
|
|
a1d982ca76 | ||
|
|
03be7f1ee3 | ||
|
|
1ced4eaf10 | ||
|
|
4970167726 | ||
|
|
065ccbc4aa | ||
|
|
1feaa63813 | ||
|
|
8541c6be35 | ||
|
|
18654526c8 | ||
|
|
bd1f7f9492 | ||
|
|
40d75dd283 | ||
|
|
f5011365f1 | ||
|
|
ebbd046c92 | ||
|
|
b9979088ec | ||
|
|
bd0e53bd2e | ||
|
|
4b586f4105 | ||
|
|
0410119d5f | ||
|
|
c2f67398d0 | ||
|
|
81149344a5 | ||
|
|
e9b649b345 | ||
|
|
9c538f4237 | ||
|
|
c08a358a2b | ||
|
|
dbaef5e79b | ||
|
|
48a155a213 | ||
|
|
c69a9a77c6 | ||
|
|
ef79fa3898 | ||
|
|
f7ad602bfc | ||
|
|
ffd6049ba2 | ||
|
|
b3ad54c0c2 | ||
|
|
67eca3f58d | ||
|
|
0847bf560f | ||
|
|
8074565c3e | ||
|
|
24e81267b9 | ||
|
|
c0ffee3ff9 | ||
|
|
c0ffeeef26 | ||
|
|
22dbf90b7d | ||
|
|
40c68c9c83 | ||
|
|
b7f0f158ab | ||
|
|
d8f31046a9 | ||
|
|
6c6382ed69 | ||
|
|
53ba1372f9 | ||
|
|
57c9cf64eb | ||
|
|
e83b1e4159 | ||
|
|
3784020a8d | ||
|
|
1223444738 | ||
|
|
2bfeed2b00 | ||
|
|
04b800d768 | ||
|
|
f1d62978ef | ||
|
|
ffadc64f28 | ||
|
|
5302f40342 | ||
|
|
b0f8c2c03e | ||
|
|
37e40673ff | ||
|
|
fefee4c675 |
39
.github/workflows/liquid.yml
vendored
Normal file
39
.github/workflows/liquid.yml
vendored
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
name: Liquid
|
||||||
|
on: [push, pull_request]
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
entry:
|
||||||
|
- { ruby: 2.5, allowed-failure: false } # minimum supported
|
||||||
|
- { ruby: 3.0, allowed-failure: false } # latest
|
||||||
|
- { ruby: ruby-head, allowed-failure: true }
|
||||||
|
name: test (${{ matrix.entry.ruby }})
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: ruby/setup-ruby@v1
|
||||||
|
with:
|
||||||
|
ruby-version: ${{ matrix.entry.ruby }}
|
||||||
|
- uses: actions/cache@v1
|
||||||
|
with:
|
||||||
|
path: vendor/bundle
|
||||||
|
key: ${{ runner.os }}-gems-${{ hashFiles('Gemfile') }}
|
||||||
|
restore-keys: ${{ runner.os }}-gems-
|
||||||
|
- run: bundle install --jobs=3 --retry=3 --path=vendor/bundle
|
||||||
|
- run: bundle exec rake
|
||||||
|
continue-on-error: ${{ matrix.entry.allowed-failure }}
|
||||||
|
memory_profile:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: ruby/setup-ruby@v1
|
||||||
|
with:
|
||||||
|
ruby-version: 2.7
|
||||||
|
- uses: actions/cache@v1
|
||||||
|
with:
|
||||||
|
path: vendor/bundle
|
||||||
|
key: ${{ runner.os }}-gems-${{ hashFiles('Gemfile') }}
|
||||||
|
restore-keys: ${{ runner.os }}-gems-
|
||||||
|
- run: bundle install --jobs=3 --retry=3 --path=vendor/bundle
|
||||||
|
- run: bundle exec rake memory_profile:run
|
||||||
@@ -20,7 +20,7 @@ Style/Alias:
|
|||||||
- prefer_alias
|
- prefer_alias
|
||||||
- prefer_alias_method
|
- prefer_alias_method
|
||||||
|
|
||||||
Layout/AlignHash:
|
Layout/HashAlignment:
|
||||||
EnforcedHashRocketStyle: key
|
EnforcedHashRocketStyle: key
|
||||||
EnforcedColonStyle: key
|
EnforcedColonStyle: key
|
||||||
EnforcedLastArgumentHashStyle: ignore_implicit
|
EnforcedLastArgumentHashStyle: ignore_implicit
|
||||||
@@ -30,7 +30,7 @@ Layout/AlignHash:
|
|||||||
- ignore_implicit
|
- ignore_implicit
|
||||||
- ignore_explicit
|
- ignore_explicit
|
||||||
|
|
||||||
Layout/AlignParameters:
|
Layout/ParameterAlignment:
|
||||||
EnforcedStyle: with_fixed_indentation
|
EnforcedStyle: with_fixed_indentation
|
||||||
SupportedStyles:
|
SupportedStyles:
|
||||||
- with_first_parameter
|
- with_first_parameter
|
||||||
@@ -76,13 +76,6 @@ Style/BlockDelimiters:
|
|||||||
- proc
|
- proc
|
||||||
- it
|
- it
|
||||||
|
|
||||||
Style/BracesAroundHashParameters:
|
|
||||||
EnforcedStyle: no_braces
|
|
||||||
SupportedStyles:
|
|
||||||
- braces
|
|
||||||
- no_braces
|
|
||||||
- context_dependent
|
|
||||||
|
|
||||||
Layout/CaseIndentation:
|
Layout/CaseIndentation:
|
||||||
EnforcedStyle: end
|
EnforcedStyle: end
|
||||||
SupportedStyles:
|
SupportedStyles:
|
||||||
@@ -172,7 +165,7 @@ Naming/FileName:
|
|||||||
Regex:
|
Regex:
|
||||||
IgnoreExecutableScripts: true
|
IgnoreExecutableScripts: true
|
||||||
|
|
||||||
Layout/IndentFirstArgument:
|
Layout/FirstArgumentIndentation:
|
||||||
EnforcedStyle: consistent
|
EnforcedStyle: consistent
|
||||||
SupportedStyles:
|
SupportedStyles:
|
||||||
- consistent
|
- consistent
|
||||||
@@ -202,6 +195,7 @@ Style/FrozenStringLiteralComment:
|
|||||||
SupportedStyles:
|
SupportedStyles:
|
||||||
- always
|
- always
|
||||||
- never
|
- never
|
||||||
|
SafeAutoCorrect: true
|
||||||
|
|
||||||
Style/GlobalVars:
|
Style/GlobalVars:
|
||||||
AllowedVariables: []
|
AllowedVariables: []
|
||||||
@@ -225,7 +219,7 @@ Layout/IndentationConsistency:
|
|||||||
Layout/IndentationWidth:
|
Layout/IndentationWidth:
|
||||||
Width: 2
|
Width: 2
|
||||||
|
|
||||||
Layout/IndentFirstArrayElement:
|
Layout/FirstArrayElementIndentation:
|
||||||
EnforcedStyle: consistent
|
EnforcedStyle: consistent
|
||||||
SupportedStyles:
|
SupportedStyles:
|
||||||
- special_inside_parentheses
|
- special_inside_parentheses
|
||||||
@@ -233,10 +227,10 @@ Layout/IndentFirstArrayElement:
|
|||||||
- align_brackets
|
- align_brackets
|
||||||
IndentationWidth:
|
IndentationWidth:
|
||||||
|
|
||||||
Layout/IndentAssignment:
|
Layout/AssignmentIndentation:
|
||||||
IndentationWidth:
|
IndentationWidth:
|
||||||
|
|
||||||
Layout/IndentFirstHashElement:
|
Layout/FirstHashElementIndentation:
|
||||||
EnforcedStyle: consistent
|
EnforcedStyle: consistent
|
||||||
SupportedStyles:
|
SupportedStyles:
|
||||||
- special_inside_parentheses
|
- special_inside_parentheses
|
||||||
@@ -271,7 +265,7 @@ Style/MethodCallWithArgsParentheses:
|
|||||||
- raise
|
- raise
|
||||||
- puts
|
- puts
|
||||||
Exclude:
|
Exclude:
|
||||||
- Gemfile
|
- '**/Gemfile'
|
||||||
|
|
||||||
Style/MethodDefParentheses:
|
Style/MethodDefParentheses:
|
||||||
EnforcedStyle: require_parentheses
|
EnforcedStyle: require_parentheses
|
||||||
@@ -340,9 +334,9 @@ Style/PercentQLiterals:
|
|||||||
Naming/PredicateName:
|
Naming/PredicateName:
|
||||||
NamePrefix:
|
NamePrefix:
|
||||||
- is_
|
- is_
|
||||||
NamePrefixBlacklist:
|
ForbiddenPrefixes:
|
||||||
- is_
|
- is_
|
||||||
NameWhitelist:
|
AllowedMethods:
|
||||||
- is_a?
|
- is_a?
|
||||||
Exclude:
|
Exclude:
|
||||||
- 'spec/**/*'
|
- 'spec/**/*'
|
||||||
@@ -467,7 +461,7 @@ Style/TernaryParentheses:
|
|||||||
- require_no_parentheses
|
- require_no_parentheses
|
||||||
AllowSafeAssignment: true
|
AllowSafeAssignment: true
|
||||||
|
|
||||||
Layout/TrailingBlankLines:
|
Layout/TrailingEmptyLines:
|
||||||
EnforcedStyle: final_newline
|
EnforcedStyle: final_newline
|
||||||
SupportedStyles:
|
SupportedStyles:
|
||||||
- final_newline
|
- final_newline
|
||||||
@@ -478,7 +472,7 @@ Style/TrivialAccessors:
|
|||||||
AllowPredicates: true
|
AllowPredicates: true
|
||||||
AllowDSLWriters: false
|
AllowDSLWriters: false
|
||||||
IgnoreClassMethods: false
|
IgnoreClassMethods: false
|
||||||
Whitelist:
|
AllowedMethods:
|
||||||
- to_ary
|
- to_ary
|
||||||
- to_a
|
- to_a
|
||||||
- to_c
|
- to_c
|
||||||
@@ -509,7 +503,7 @@ Style/WhileUntilModifier:
|
|||||||
Metrics/BlockNesting:
|
Metrics/BlockNesting:
|
||||||
Max: 3
|
Max: 3
|
||||||
|
|
||||||
Metrics/LineLength:
|
Layout/LineLength:
|
||||||
Max: 120
|
Max: 120
|
||||||
AllowHeredoc: true
|
AllowHeredoc: true
|
||||||
AllowURI: true
|
AllowURI: true
|
||||||
@@ -561,7 +555,7 @@ Lint/UnusedMethodArgument:
|
|||||||
Naming/AccessorMethodName:
|
Naming/AccessorMethodName:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Layout/AlignArray:
|
Layout/ArrayAlignment:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/ArrayJoin:
|
Style/ArrayJoin:
|
||||||
@@ -584,6 +578,7 @@ Layout/BlockEndNewline:
|
|||||||
|
|
||||||
Style/CaseEquality:
|
Style/CaseEquality:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
AllowOnConstant: true
|
||||||
|
|
||||||
Style/CharacterLiteral:
|
Style/CharacterLiteral:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
@@ -666,6 +661,9 @@ Style/IfWithSemicolon:
|
|||||||
Style/IdenticalConditionalBranches:
|
Style/IdenticalConditionalBranches:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
|
Layout/IndentationStyle:
|
||||||
|
Enabled: true
|
||||||
|
|
||||||
Style/InfiniteLoop:
|
Style/InfiniteLoop:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
@@ -678,7 +676,7 @@ Style/LineEndConcatenation:
|
|||||||
Style/MethodCallWithoutArgsParentheses:
|
Style/MethodCallWithoutArgsParentheses:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/MethodMissingSuper:
|
Lint/MissingSuper:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/MissingRespondToMissing:
|
Style/MissingRespondToMissing:
|
||||||
@@ -810,22 +808,19 @@ Layout/SpaceInsideRangeLiteral:
|
|||||||
Style/SymbolLiteral:
|
Style/SymbolLiteral:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Layout/Tab:
|
|
||||||
Enabled: true
|
|
||||||
|
|
||||||
Layout/TrailingWhitespace:
|
Layout/TrailingWhitespace:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/UnlessElse:
|
Style/UnlessElse:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/UnneededCapitalW:
|
Style/RedundantCapitalW:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/UnneededInterpolation:
|
Style/RedundantInterpolation:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/UnneededPercentQ:
|
Style/RedundantPercentQ:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Style/VariableInterpolation:
|
Style/VariableInterpolation:
|
||||||
@@ -840,8 +835,8 @@ Style/WhileUntilDo:
|
|||||||
Style/ZeroLengthPredicate:
|
Style/ZeroLengthPredicate:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Layout/IndentHeredoc:
|
Layout/HeredocIndentation:
|
||||||
EnforcedStyle: squiggly
|
Enabled: true
|
||||||
|
|
||||||
Lint/AmbiguousOperator:
|
Lint/AmbiguousOperator:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
@@ -864,7 +859,7 @@ Lint/DeprecatedClassMethods:
|
|||||||
Lint/DuplicateMethods:
|
Lint/DuplicateMethods:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/DuplicatedKey:
|
Lint/DuplicateHashKey:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/EachWithObjectArgument:
|
Lint/EachWithObjectArgument:
|
||||||
@@ -879,9 +874,6 @@ Lint/EmptyEnsure:
|
|||||||
Lint/EmptyInterpolation:
|
Lint/EmptyInterpolation:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/EndInMethod:
|
|
||||||
Enabled: true
|
|
||||||
|
|
||||||
Lint/EnsureReturn:
|
Lint/EnsureReturn:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
@@ -891,8 +883,8 @@ Lint/FloatOutOfRange:
|
|||||||
Lint/FormatParameterMismatch:
|
Lint/FormatParameterMismatch:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/HandleExceptions:
|
Lint/SuppressedException:
|
||||||
Enabled: true
|
AllowComments: true
|
||||||
|
|
||||||
Lint/ImplicitStringConcatenation:
|
Lint/ImplicitStringConcatenation:
|
||||||
Description: Checks for adjacent string literals on the same line, which could
|
Description: Checks for adjacent string literals on the same line, which could
|
||||||
@@ -947,7 +939,7 @@ Lint/ShadowedException:
|
|||||||
Lint/ShadowingOuterLocalVariable:
|
Lint/ShadowingOuterLocalVariable:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/StringConversionInInterpolation:
|
Lint/RedundantStringCoercion:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/UnderscorePrefixedVariableName:
|
Lint/UnderscorePrefixedVariableName:
|
||||||
@@ -956,13 +948,13 @@ Lint/UnderscorePrefixedVariableName:
|
|||||||
Lint/UnifiedInteger:
|
Lint/UnifiedInteger:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/UnneededCopDisableDirective:
|
Lint/RedundantCopDisableDirective:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/UnneededCopEnableDirective:
|
Lint/RedundantCopEnableDirective:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/UnneededSplatExpansion:
|
Lint/RedundantSplatExpansion:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/UnreachableCode:
|
Lint/UnreachableCode:
|
||||||
@@ -974,7 +966,7 @@ Lint/UselessAccessModifier:
|
|||||||
Lint/UselessAssignment:
|
Lint/UselessAssignment:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/UselessComparison:
|
Lint/BinaryOperatorWithIdenticalOperands:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
Lint/UselessElseWithoutRescue:
|
Lint/UselessElseWithoutRescue:
|
||||||
@@ -1025,3 +1017,6 @@ Style/ModuleFunction:
|
|||||||
|
|
||||||
Lint/OrderedMagicComments:
|
Lint/OrderedMagicComments:
|
||||||
Enabled: true
|
Enabled: true
|
||||||
|
|
||||||
|
Lint/DeprecatedOpenSSLConstant:
|
||||||
|
Enabled: true
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ Performance:
|
|||||||
|
|
||||||
AllCops:
|
AllCops:
|
||||||
TargetRubyVersion: 2.4
|
TargetRubyVersion: 2.4
|
||||||
|
NewCops: disable
|
||||||
Exclude:
|
Exclude:
|
||||||
- 'vendor/bundle/**/*'
|
- 'vendor/bundle/**/*'
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# This configuration was generated by
|
# This configuration was generated by
|
||||||
# `rubocop --auto-gen-config`
|
# `rubocop --auto-gen-config`
|
||||||
# on 2019-09-11 06:34:25 +1000 using RuboCop version 0.74.0.
|
# on 2020-12-11 18:53:41 UTC using RuboCop version 1.6.1.
|
||||||
# The point is for the user to remove these configuration records
|
# The point is for the user to remove these configuration records
|
||||||
# one by one as the offenses are removed from the code base.
|
# one by one as the offenses are removed from the code base.
|
||||||
# Note that changes in the inspected code, or installation of new
|
# Note that changes in the inspected code, or installation of new
|
||||||
@@ -14,14 +14,26 @@ Lint/InheritException:
|
|||||||
Exclude:
|
Exclude:
|
||||||
- 'lib/liquid/interrupts.rb'
|
- 'lib/liquid/interrupts.rb'
|
||||||
|
|
||||||
# Offense count: 98
|
# Offense count: 113
|
||||||
# Cop supports --auto-correct.
|
# Cop supports --auto-correct.
|
||||||
# Configuration parameters: AutoCorrect, AllowHeredoc, AllowURI, URISchemes, IgnoreCopDirectives, IgnoredPatterns.
|
# Configuration parameters: AutoCorrect, AllowHeredoc, AllowURI, URISchemes, IgnoreCopDirectives, IgnoredPatterns.
|
||||||
# URISchemes: http, https
|
# URISchemes: http, https
|
||||||
Metrics/LineLength:
|
Layout/LineLength:
|
||||||
Max: 294
|
Max: 260
|
||||||
|
|
||||||
# Offense count: 44
|
# Offense count: 8
|
||||||
|
Lint/MissingSuper:
|
||||||
|
Exclude:
|
||||||
|
- 'lib/liquid/forloop_drop.rb'
|
||||||
|
- 'lib/liquid/tablerowloop_drop.rb'
|
||||||
|
- 'test/integration/assign_test.rb'
|
||||||
|
- 'test/integration/context_test.rb'
|
||||||
|
- 'test/integration/filter_test.rb'
|
||||||
|
- 'test/integration/standard_filter_test.rb'
|
||||||
|
- 'test/integration/tags/for_tag_test.rb'
|
||||||
|
- 'test/integration/tags/table_row_test.rb'
|
||||||
|
|
||||||
|
# Offense count: 43
|
||||||
Naming/ConstantName:
|
Naming/ConstantName:
|
||||||
Exclude:
|
Exclude:
|
||||||
- 'lib/liquid.rb'
|
- 'lib/liquid.rb'
|
||||||
@@ -32,7 +44,6 @@ Naming/ConstantName:
|
|||||||
- 'lib/liquid/tags/cycle.rb'
|
- 'lib/liquid/tags/cycle.rb'
|
||||||
- 'lib/liquid/tags/for.rb'
|
- 'lib/liquid/tags/for.rb'
|
||||||
- 'lib/liquid/tags/if.rb'
|
- 'lib/liquid/tags/if.rb'
|
||||||
- 'lib/liquid/tags/include.rb'
|
|
||||||
- 'lib/liquid/tags/raw.rb'
|
- 'lib/liquid/tags/raw.rb'
|
||||||
- 'lib/liquid/tags/table_row.rb'
|
- 'lib/liquid/tags/table_row.rb'
|
||||||
- 'lib/liquid/variable.rb'
|
- 'lib/liquid/variable.rb'
|
||||||
@@ -40,9 +51,7 @@ Naming/ConstantName:
|
|||||||
- 'performance/shopify/paginate.rb'
|
- 'performance/shopify/paginate.rb'
|
||||||
- 'test/integration/tags/include_tag_test.rb'
|
- 'test/integration/tags/include_tag_test.rb'
|
||||||
|
|
||||||
# Offense count: 5
|
# Offense count: 2
|
||||||
Style/ClassVars:
|
Style/ClassVars:
|
||||||
Exclude:
|
Exclude:
|
||||||
- 'lib/liquid/condition.rb'
|
- 'lib/liquid/condition.rb'
|
||||||
- 'lib/liquid/strainer.rb'
|
|
||||||
- 'lib/liquid/template.rb'
|
|
||||||
26
.travis.yml
26
.travis.yml
@@ -1,26 +0,0 @@
|
|||||||
language: ruby
|
|
||||||
cache: bundler
|
|
||||||
|
|
||||||
rvm:
|
|
||||||
- 2.4
|
|
||||||
- 2.5
|
|
||||||
- &latest_ruby 2.6
|
|
||||||
- 2.7
|
|
||||||
- ruby-head
|
|
||||||
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
- rvm: *latest_ruby
|
|
||||||
script: bundle exec rake memory_profile:run
|
|
||||||
name: Profiling Memory Usage
|
|
||||||
allow_failures:
|
|
||||||
- rvm: ruby-head
|
|
||||||
|
|
||||||
branches:
|
|
||||||
only:
|
|
||||||
- master
|
|
||||||
- gh-pages
|
|
||||||
- /.*-stable/
|
|
||||||
|
|
||||||
notifications:
|
|
||||||
disable: true
|
|
||||||
4
Gemfile
4
Gemfile
@@ -18,10 +18,10 @@ group :benchmark, :test do
|
|||||||
end
|
end
|
||||||
|
|
||||||
group :test do
|
group :test do
|
||||||
gem 'rubocop', '~> 0.74.0', require: false
|
gem 'rubocop', '~> 1.4', require: false
|
||||||
gem 'rubocop-performance', require: false
|
gem 'rubocop-performance', require: false
|
||||||
|
|
||||||
platform :mri, :truffleruby do
|
platform :mri, :truffleruby do
|
||||||
gem 'liquid-c', github: 'Shopify/liquid-c', ref: 'master'
|
gem 'liquid-c', github: 'Shopify/liquid-c', ref: 'inline-comment'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
38
History.md
38
History.md
@@ -1,5 +1,43 @@
|
|||||||
# Liquid Change Log
|
# Liquid Change Log
|
||||||
|
|
||||||
|
## Unreleased
|
||||||
|
|
||||||
|
### Features
|
||||||
|
* Allow `#` to be used as an inline comment tag (#1401) [Dylan Thacker-Smith]
|
||||||
|
|
||||||
|
## 5.0.0 / 2021-01-06
|
||||||
|
|
||||||
|
### Features
|
||||||
|
* Add new `{% render %}` tag (#1122) [Samuel Doiron]
|
||||||
|
* Add support for `as` in `{% render %}` and `{% include %}` (#1181) [Mike Angell]
|
||||||
|
* Add `{% liquid %}` and `{% echo %}` tags (#1086) [Justin Li]
|
||||||
|
* Add [usage tracking](README.md#usage-tracking) [Mike Angell]
|
||||||
|
* Add `Tag.disable_tags` for disabling tags that prepend `Tag::Disableable` at render time (#1162, #1274, #1275) [Mike Angell]
|
||||||
|
* Support using a profiler for multiple renders (#1365, #1366) [Dylan Thacker-Smith]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
* Fix catastrophic backtracking in `RANGES_REGEX` regular expression (#1357) [Dylan Thacker-Smith]
|
||||||
|
* Make sure the for tag's limit and offset are integers (#1094) [David Cornu]
|
||||||
|
* Invokable methods for enumerable reject include (#1151) [Thierry Joyal]
|
||||||
|
* Allow `default` filter to handle `false` as value (#1144) [Mike Angell]
|
||||||
|
* Fix render length resource limit so it doesn't multiply nested output (#1285) [Dylan Thacker-Smith]
|
||||||
|
* Fix duplication of text in raw tags (#1304) [Peter Zhu]
|
||||||
|
* Fix strict parsing of find variable with a name expression (#1317) [Dylan Thacker-Smith]
|
||||||
|
* Use monotonic time to measure durations in Liquid::Profiler (#1362) [Dylan Thacker-Smith]
|
||||||
|
|
||||||
|
### Breaking Changes
|
||||||
|
* Require Ruby >= 2.5 (#1131, #1310) [Mike Angell, Dylan Thacker-Smith]
|
||||||
|
* Remove support for taint checking (#1268) [Dylan Thacker-Smith]
|
||||||
|
* Split Strainer class into StrainerFactory and StrainerTemplate (#1208) [Thierry Joyal]
|
||||||
|
* Remove handling of a nil context in the Strainer class (#1218) [Thierry Joyal]
|
||||||
|
* Handle `BlockBody#blank?` at parse time (#1287) [Dylan Thacker-Smith]
|
||||||
|
* Pass the tag markup and tokenizer to `Document#unknown_tag` (#1290) [Dylan Thacker-Smith]
|
||||||
|
* And several internal changes
|
||||||
|
|
||||||
|
### Performance Improvements
|
||||||
|
* Reduce allocations (#1073, #1091, #1115, #1099, #1117, #1141, #1322, #1341) [Richard Monette, Florian Weingarten, Ashwin Maroli]
|
||||||
|
* Improve resources limits performance (#1093, #1323) [Florian Weingarten, Dylan Thacker-Smith]
|
||||||
|
|
||||||
## 4.0.3 / 2019-03-12
|
## 4.0.3 / 2019-03-12
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|||||||
16
Rakefile
16
Rakefile
@@ -9,11 +9,17 @@ task(default: [:test, :rubocop])
|
|||||||
|
|
||||||
desc('run test suite with default parser')
|
desc('run test suite with default parser')
|
||||||
Rake::TestTask.new(:base_test) do |t|
|
Rake::TestTask.new(:base_test) do |t|
|
||||||
t.libs << '.' << 'lib' << 'test'
|
t.libs << 'lib' << 'test'
|
||||||
t.test_files = FileList['test/{integration,unit}/**/*_test.rb']
|
t.test_files = FileList['test/{integration,unit}/**/*_test.rb']
|
||||||
t.verbose = false
|
t.verbose = false
|
||||||
end
|
end
|
||||||
|
|
||||||
|
Rake::TestTask.new(:integration_test) do |t|
|
||||||
|
t.libs << 'lib' << 'test'
|
||||||
|
t.test_files = FileList['test/integration/**/*_test.rb']
|
||||||
|
t.verbose = false
|
||||||
|
end
|
||||||
|
|
||||||
desc('run test suite with warn error mode')
|
desc('run test suite with warn error mode')
|
||||||
task :warn_test do
|
task :warn_test do
|
||||||
ENV['LIQUID_PARSER_MODE'] = 'warn'
|
ENV['LIQUID_PARSER_MODE'] = 'warn'
|
||||||
@@ -40,12 +46,12 @@ task :test do
|
|||||||
ENV['LIQUID_C'] = '1'
|
ENV['LIQUID_C'] = '1'
|
||||||
|
|
||||||
ENV['LIQUID_PARSER_MODE'] = 'lax'
|
ENV['LIQUID_PARSER_MODE'] = 'lax'
|
||||||
Rake::Task['base_test'].reenable
|
Rake::Task['integration_test'].reenable
|
||||||
Rake::Task['base_test'].invoke
|
Rake::Task['integration_test'].invoke
|
||||||
|
|
||||||
ENV['LIQUID_PARSER_MODE'] = 'strict'
|
ENV['LIQUID_PARSER_MODE'] = 'strict'
|
||||||
Rake::Task['base_test'].reenable
|
Rake::Task['integration_test'].reenable
|
||||||
Rake::Task['base_test'].invoke
|
Rake::Task['integration_test'].invoke
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -42,6 +42,8 @@ module Liquid
|
|||||||
TemplateParser = /(#{PartialTemplateParser}|#{AnyStartingTag})/om
|
TemplateParser = /(#{PartialTemplateParser}|#{AnyStartingTag})/om
|
||||||
VariableParser = /\[[^\]]+\]|#{VariableSegment}+\??/o
|
VariableParser = /\[[^\]]+\]|#{VariableSegment}+\??/o
|
||||||
|
|
||||||
|
RAISE_EXCEPTION_LAMBDA = ->(_e) { raise }
|
||||||
|
|
||||||
singleton_class.send(:attr_accessor, :cache_classes)
|
singleton_class.send(:attr_accessor, :cache_classes)
|
||||||
self.cache_classes = true
|
self.cache_classes = true
|
||||||
end
|
end
|
||||||
@@ -57,11 +59,14 @@ require 'liquid/forloop_drop'
|
|||||||
require 'liquid/extensions'
|
require 'liquid/extensions'
|
||||||
require 'liquid/errors'
|
require 'liquid/errors'
|
||||||
require 'liquid/interrupts'
|
require 'liquid/interrupts'
|
||||||
require 'liquid/strainer'
|
require 'liquid/strainer_factory'
|
||||||
|
require 'liquid/strainer_template'
|
||||||
require 'liquid/expression'
|
require 'liquid/expression'
|
||||||
require 'liquid/context'
|
require 'liquid/context'
|
||||||
require 'liquid/parser_switching'
|
require 'liquid/parser_switching'
|
||||||
require 'liquid/tag'
|
require 'liquid/tag'
|
||||||
|
require 'liquid/tag/disabler'
|
||||||
|
require 'liquid/tag/disableable'
|
||||||
require 'liquid/block'
|
require 'liquid/block'
|
||||||
require 'liquid/block_body'
|
require 'liquid/block_body'
|
||||||
require 'liquid/document'
|
require 'liquid/document'
|
||||||
@@ -80,8 +85,8 @@ require 'liquid/partial_cache'
|
|||||||
require 'liquid/usage'
|
require 'liquid/usage'
|
||||||
require 'liquid/register'
|
require 'liquid/register'
|
||||||
require 'liquid/static_registers'
|
require 'liquid/static_registers'
|
||||||
|
require 'liquid/template_factory'
|
||||||
|
|
||||||
# Load all the tags of the standard library
|
# Load all the tags of the standard library
|
||||||
#
|
#
|
||||||
Dir["#{__dir__}/liquid/tags/*.rb"].each { |f| require f }
|
Dir["#{__dir__}/liquid/tags/*.rb"].each { |f| require f }
|
||||||
Dir["#{__dir__}/liquid/registers/*.rb"].each { |f| require f }
|
|
||||||
|
|||||||
@@ -10,9 +10,10 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def parse(tokens)
|
def parse(tokens)
|
||||||
@body = BlockBody.new
|
@body = new_body
|
||||||
while parse_body(@body, tokens)
|
while parse_body(@body, tokens)
|
||||||
end
|
end
|
||||||
|
@body.freeze
|
||||||
end
|
end
|
||||||
|
|
||||||
# For backwards compatibility
|
# For backwards compatibility
|
||||||
@@ -28,7 +29,12 @@ module Liquid
|
|||||||
@body.nodelist
|
@body.nodelist
|
||||||
end
|
end
|
||||||
|
|
||||||
def unknown_tag(tag, _params, _tokens)
|
def unknown_tag(tag_name, _markup, _tokenizer)
|
||||||
|
Block.raise_unknown_tag(tag_name, block_name, block_delimiter, parse_context)
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api private
|
||||||
|
def self.raise_unknown_tag(tag, block_name, block_delimiter, parse_context)
|
||||||
if tag == 'else'
|
if tag == 'else'
|
||||||
raise SyntaxError, parse_context.locale.t("errors.syntax.unexpected_else",
|
raise SyntaxError, parse_context.locale.t("errors.syntax.unexpected_else",
|
||||||
block_name: block_name)
|
block_name: block_name)
|
||||||
@@ -42,6 +48,10 @@ module Liquid
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def raise_tag_never_closed(block_name)
|
||||||
|
raise SyntaxError, parse_context.locale.t("errors.syntax.tag_never_closed", block_name: block_name)
|
||||||
|
end
|
||||||
|
|
||||||
def block_name
|
def block_name
|
||||||
@tag_name
|
@tag_name
|
||||||
end
|
end
|
||||||
@@ -50,8 +60,14 @@ module Liquid
|
|||||||
@block_delimiter ||= "end#{block_name}"
|
@block_delimiter ||= "end#{block_name}"
|
||||||
end
|
end
|
||||||
|
|
||||||
protected
|
private
|
||||||
|
|
||||||
|
# @api public
|
||||||
|
def new_body
|
||||||
|
parse_context.new_block_body
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api public
|
||||||
def parse_body(body, tokens)
|
def parse_body(body, tokens)
|
||||||
if parse_context.depth >= MAX_DEPTH
|
if parse_context.depth >= MAX_DEPTH
|
||||||
raise StackLevelError, "Nesting too deep"
|
raise StackLevelError, "Nesting too deep"
|
||||||
@@ -62,9 +78,7 @@ module Liquid
|
|||||||
@blank &&= body.blank?
|
@blank &&= body.blank?
|
||||||
|
|
||||||
return false if end_tag_name == block_delimiter
|
return false if end_tag_name == block_delimiter
|
||||||
unless end_tag_name
|
raise_tag_never_closed(block_name) unless end_tag_name
|
||||||
raise SyntaxError, parse_context.locale.t("errors.syntax.tag_never_closed", block_name: block_name)
|
|
||||||
end
|
|
||||||
|
|
||||||
# this tag is not registered with the system
|
# this tag is not registered with the system
|
||||||
# pass it to the current block for special handling or error reporting
|
# pass it to the current block for special handling or error reporting
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'English'
|
||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
class BlockBody
|
class BlockBody
|
||||||
LiquidTagToken = /\A\s*(\w+)\s*(.*?)\z/o
|
LiquidTagToken = /\A\s*(\w+|#)\s*(.*?)\z/o
|
||||||
FullToken = /\A#{TagStart}#{WhitespaceControl}?(\s*)(\w+)(\s*)(.*?)#{WhitespaceControl}?#{TagEnd}\z/om
|
FullToken = /\A#{TagStart}#{WhitespaceControl}?(\s*)(\w+|#)(\s*)(.*?)#{WhitespaceControl}?#{TagEnd}\z/om
|
||||||
ContentOfVariable = /\A#{VariableStart}#{WhitespaceControl}?(.*?)#{WhitespaceControl}?#{VariableEnd}\z/om
|
ContentOfVariable = /\A#{VariableStart}#{WhitespaceControl}?(.*?)#{WhitespaceControl}?#{VariableEnd}\z/om
|
||||||
WhitespaceOrNothing = /\A\s*\z/
|
WhitespaceOrNothing = /\A\s*\z/
|
||||||
TAGSTART = "{%"
|
TAGSTART = "{%"
|
||||||
@@ -17,6 +19,8 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def parse(tokenizer, parse_context, &block)
|
def parse(tokenizer, parse_context, &block)
|
||||||
|
raise FrozenError, "can't modify frozen Liquid::BlockBody" if frozen?
|
||||||
|
|
||||||
parse_context.line_number = tokenizer.line_number
|
parse_context.line_number = tokenizer.line_number
|
||||||
|
|
||||||
if tokenizer.for_liquid_tag
|
if tokenizer.for_liquid_tag
|
||||||
@@ -26,6 +30,11 @@ module Liquid
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def freeze
|
||||||
|
@nodelist.freeze
|
||||||
|
super
|
||||||
|
end
|
||||||
|
|
||||||
private def parse_for_liquid_tag(tokenizer, parse_context)
|
private def parse_for_liquid_tag(tokenizer, parse_context)
|
||||||
while (token = tokenizer.shift)
|
while (token = tokenizer.shift)
|
||||||
unless token.empty? || token =~ WhitespaceOrNothing
|
unless token.empty? || token =~ WhitespaceOrNothing
|
||||||
@@ -51,14 +60,63 @@ module Liquid
|
|||||||
yield nil, nil
|
yield nil, nil
|
||||||
end
|
end
|
||||||
|
|
||||||
private def parse_for_document(tokenizer, parse_context, &block)
|
# @api private
|
||||||
|
def self.unknown_tag_in_liquid_tag(tag, parse_context)
|
||||||
|
Block.raise_unknown_tag(tag, 'liquid', '%}', parse_context)
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api private
|
||||||
|
def self.raise_missing_tag_terminator(token, parse_context)
|
||||||
|
raise SyntaxError, parse_context.locale.t("errors.syntax.tag_termination", token: token, tag_end: TagEnd.inspect)
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api private
|
||||||
|
def self.raise_missing_variable_terminator(token, parse_context)
|
||||||
|
raise SyntaxError, parse_context.locale.t("errors.syntax.variable_termination", token: token, tag_end: VariableEnd.inspect)
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api private
|
||||||
|
def self.render_node(context, output, node)
|
||||||
|
node.render_to_output_buffer(context, output)
|
||||||
|
rescue => exc
|
||||||
|
blank_tag = !node.instance_of?(Variable) && node.blank?
|
||||||
|
rescue_render_node(context, output, node.line_number, exc, blank_tag)
|
||||||
|
end
|
||||||
|
|
||||||
|
# @api private
|
||||||
|
def self.rescue_render_node(context, output, line_number, exc, blank_tag)
|
||||||
|
case exc
|
||||||
|
when MemoryError
|
||||||
|
raise
|
||||||
|
when UndefinedVariable, UndefinedDropMethod, UndefinedFilter
|
||||||
|
context.handle_error(exc, line_number)
|
||||||
|
else
|
||||||
|
error_message = context.handle_error(exc, line_number)
|
||||||
|
unless blank_tag # conditional for backwards compatibility
|
||||||
|
output << error_message
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private def parse_liquid_tag(markup, parse_context)
|
||||||
|
liquid_tag_tokenizer = parse_context.new_tokenizer(
|
||||||
|
markup, start_line_number: parse_context.line_number, for_liquid_tag: true
|
||||||
|
)
|
||||||
|
parse_for_liquid_tag(liquid_tag_tokenizer, parse_context) do |end_tag_name, _end_tag_markup|
|
||||||
|
if end_tag_name
|
||||||
|
BlockBody.unknown_tag_in_liquid_tag(end_tag_name, parse_context)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private def parse_for_document(tokenizer, parse_context)
|
||||||
while (token = tokenizer.shift)
|
while (token = tokenizer.shift)
|
||||||
next if token.empty?
|
next if token.empty?
|
||||||
case
|
case
|
||||||
when token.start_with?(TAGSTART)
|
when token.start_with?(TAGSTART)
|
||||||
whitespace_handler(token, parse_context)
|
whitespace_handler(token, parse_context)
|
||||||
unless token =~ FullToken
|
unless token =~ FullToken
|
||||||
raise_missing_tag_terminator(token, parse_context)
|
BlockBody.raise_missing_tag_terminator(token, parse_context)
|
||||||
end
|
end
|
||||||
tag_name = Regexp.last_match(2)
|
tag_name = Regexp.last_match(2)
|
||||||
markup = Regexp.last_match(4)
|
markup = Regexp.last_match(4)
|
||||||
@@ -70,8 +128,8 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
if tag_name == 'liquid'
|
if tag_name == 'liquid'
|
||||||
liquid_tag_tokenizer = Tokenizer.new(markup, line_number: parse_context.line_number, for_liquid_tag: true)
|
parse_liquid_tag(markup, parse_context)
|
||||||
next parse_for_liquid_tag(liquid_tag_tokenizer, parse_context, &block)
|
next
|
||||||
end
|
end
|
||||||
|
|
||||||
unless (tag = registered_tags[tag_name])
|
unless (tag = registered_tags[tag_name])
|
||||||
@@ -104,7 +162,11 @@ module Liquid
|
|||||||
if token[2] == WhitespaceControl
|
if token[2] == WhitespaceControl
|
||||||
previous_token = @nodelist.last
|
previous_token = @nodelist.last
|
||||||
if previous_token.is_a?(String)
|
if previous_token.is_a?(String)
|
||||||
|
first_byte = previous_token.getbyte(0)
|
||||||
previous_token.rstrip!
|
previous_token.rstrip!
|
||||||
|
if previous_token.empty? && parse_context[:bug_compatible_whitespace_trimming] && first_byte
|
||||||
|
previous_token << first_byte
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
parse_context.trim_whitespace = (token[-3] == WhitespaceControl)
|
parse_context.trim_whitespace = (token[-3] == WhitespaceControl)
|
||||||
@@ -114,38 +176,49 @@ module Liquid
|
|||||||
@blank
|
@blank
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Remove blank strings in the block body for a control flow tag (e.g. `if`, `for`, `case`, `unless`)
|
||||||
|
# with a blank body.
|
||||||
|
#
|
||||||
|
# For example, in a conditional assignment like the following
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# {% if size > max_size %}
|
||||||
|
# {% assign size = max_size %}
|
||||||
|
# {% endif %}
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# we assume the intention wasn't to output the blank spaces in the `if` tag's block body, so this method
|
||||||
|
# will remove them to reduce the render output size.
|
||||||
|
#
|
||||||
|
# Note that it is now preferred to use the `liquid` tag for this use case.
|
||||||
|
def remove_blank_strings
|
||||||
|
raise "remove_blank_strings only support being called on a blank block body" unless @blank
|
||||||
|
@nodelist.reject! { |node| node.instance_of?(String) }
|
||||||
|
end
|
||||||
|
|
||||||
def render(context)
|
def render(context)
|
||||||
render_to_output_buffer(context, +'')
|
render_to_output_buffer(context, +'')
|
||||||
end
|
end
|
||||||
|
|
||||||
def render_to_output_buffer(context, output)
|
def render_to_output_buffer(context, output)
|
||||||
context.resource_limits.render_score += @nodelist.length
|
freeze unless frozen?
|
||||||
|
|
||||||
|
context.resource_limits.increment_render_score(@nodelist.length)
|
||||||
|
|
||||||
idx = 0
|
idx = 0
|
||||||
while (node = @nodelist[idx])
|
while (node = @nodelist[idx])
|
||||||
previous_output_size = output.bytesize
|
if node.instance_of?(String)
|
||||||
|
|
||||||
case node
|
|
||||||
when String
|
|
||||||
output << node
|
output << node
|
||||||
when Variable
|
else
|
||||||
render_node(context, output, node)
|
render_node(context, output, node)
|
||||||
when Block
|
|
||||||
render_node(context, node.blank? ? +'' : output, node)
|
|
||||||
break if context.interrupt? # might have happened in a for-block
|
|
||||||
when Continue, Break
|
|
||||||
# If we get an Interrupt that means the block must stop processing. An
|
# If we get an Interrupt that means the block must stop processing. An
|
||||||
# Interrupt is any command that stops block execution such as {% break %}
|
# Interrupt is any command that stops block execution such as {% break %}
|
||||||
# or {% continue %}
|
# or {% continue %}. These tags may also occur through Block or Include tags.
|
||||||
context.push_interrupt(node.interrupt)
|
break if context.interrupt? # might have happened in a for-block
|
||||||
break
|
|
||||||
else # Other non-Block tags
|
|
||||||
render_node(context, output, node)
|
|
||||||
break if context.interrupt? # might have happened through an include
|
|
||||||
end
|
end
|
||||||
idx += 1
|
idx += 1
|
||||||
|
|
||||||
raise_if_resource_limits_reached(context, output.bytesize - previous_output_size)
|
context.resource_limits.increment_write_score(output)
|
||||||
end
|
end
|
||||||
|
|
||||||
output
|
output
|
||||||
@@ -154,29 +227,7 @@ module Liquid
|
|||||||
private
|
private
|
||||||
|
|
||||||
def render_node(context, output, node)
|
def render_node(context, output, node)
|
||||||
if node.disabled?(context)
|
BlockBody.render_node(context, output, node)
|
||||||
output << node.disabled_error_message
|
|
||||||
return
|
|
||||||
end
|
|
||||||
disable_tags(context, node.disabled_tags) do
|
|
||||||
node.render_to_output_buffer(context, output)
|
|
||||||
end
|
|
||||||
rescue UndefinedVariable, UndefinedDropMethod, UndefinedFilter => e
|
|
||||||
context.handle_error(e, node.line_number)
|
|
||||||
rescue ::StandardError => e
|
|
||||||
line_number = node.is_a?(String) ? nil : node.line_number
|
|
||||||
output << context.handle_error(e, line_number)
|
|
||||||
end
|
|
||||||
|
|
||||||
def disable_tags(context, tags, &block)
|
|
||||||
return yield if tags.empty?
|
|
||||||
context.registers[:disabled_tags].disable(tags, &block)
|
|
||||||
end
|
|
||||||
|
|
||||||
def raise_if_resource_limits_reached(context, length)
|
|
||||||
context.resource_limits.render_length += length
|
|
||||||
return unless context.resource_limits.reached?
|
|
||||||
raise MemoryError, "Memory limits exceeded"
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def create_variable(token, parse_context)
|
def create_variable(token, parse_context)
|
||||||
@@ -184,15 +235,17 @@ module Liquid
|
|||||||
markup = content.first
|
markup = content.first
|
||||||
return Variable.new(markup, parse_context)
|
return Variable.new(markup, parse_context)
|
||||||
end
|
end
|
||||||
raise_missing_variable_terminator(token, parse_context)
|
BlockBody.raise_missing_variable_terminator(token, parse_context)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @deprecated Use {.raise_missing_tag_terminator} instead
|
||||||
def raise_missing_tag_terminator(token, parse_context)
|
def raise_missing_tag_terminator(token, parse_context)
|
||||||
raise SyntaxError, parse_context.locale.t("errors.syntax.tag_termination", token: token, tag_end: TagEnd.inspect)
|
BlockBody.raise_missing_tag_terminator(token, parse_context)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# @deprecated Use {.raise_missing_variable_terminator} instead
|
||||||
def raise_missing_variable_terminator(token, parse_context)
|
def raise_missing_variable_terminator(token, parse_context)
|
||||||
raise SyntaxError, parse_context.locale.t("errors.syntax.variable_termination", token: token, tag_end: VariableEnd.inspect)
|
BlockBody.raise_missing_variable_terminator(token, parse_context)
|
||||||
end
|
end
|
||||||
|
|
||||||
def registered_tags
|
def registered_tags
|
||||||
|
|||||||
@@ -27,10 +27,28 @@ module Liquid
|
|||||||
end,
|
end,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class MethodLiteral
|
||||||
|
attr_reader :method_name, :to_s
|
||||||
|
|
||||||
|
def initialize(method_name, to_s)
|
||||||
|
@method_name = method_name
|
||||||
|
@to_s = to_s
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
@@method_literals = {
|
||||||
|
'blank' => MethodLiteral.new(:blank?, '').freeze,
|
||||||
|
'empty' => MethodLiteral.new(:empty?, '').freeze,
|
||||||
|
}
|
||||||
|
|
||||||
def self.operators
|
def self.operators
|
||||||
@@operators
|
@@operators
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def self.parse_expression(parse_context, markup)
|
||||||
|
@@method_literals[markup] || parse_context.parse_expression(markup)
|
||||||
|
end
|
||||||
|
|
||||||
attr_reader :attachment, :child_condition
|
attr_reader :attachment, :child_condition
|
||||||
attr_accessor :left, :operator, :right
|
attr_accessor :left, :operator, :right
|
||||||
|
|
||||||
@@ -38,6 +56,7 @@ module Liquid
|
|||||||
@left = left
|
@left = left
|
||||||
@operator = operator
|
@operator = operator
|
||||||
@right = right
|
@right = right
|
||||||
|
|
||||||
@child_relation = nil
|
@child_relation = nil
|
||||||
@child_condition = nil
|
@child_condition = nil
|
||||||
end
|
end
|
||||||
@@ -90,7 +109,7 @@ module Liquid
|
|||||||
private
|
private
|
||||||
|
|
||||||
def equal_variables(left, right)
|
def equal_variables(left, right)
|
||||||
if left.is_a?(Liquid::Expression::MethodLiteral)
|
if left.is_a?(MethodLiteral)
|
||||||
if right.respond_to?(left.method_name)
|
if right.respond_to?(left.method_name)
|
||||||
return right.send(left.method_name)
|
return right.send(left.method_name)
|
||||||
else
|
else
|
||||||
@@ -98,7 +117,7 @@ module Liquid
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if right.is_a?(Liquid::Expression::MethodLiteral)
|
if right.is_a?(MethodLiteral)
|
||||||
if left.respond_to?(right.method_name)
|
if left.respond_to?(right.method_name)
|
||||||
return left.send(right.method_name)
|
return left.send(right.method_name)
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -18,8 +18,8 @@ module Liquid
|
|||||||
attr_accessor :exception_renderer, :template_name, :partial, :global_filter, :strict_variables, :strict_filters
|
attr_accessor :exception_renderer, :template_name, :partial, :global_filter, :strict_variables, :strict_filters
|
||||||
|
|
||||||
# rubocop:disable Metrics/ParameterLists
|
# rubocop:disable Metrics/ParameterLists
|
||||||
def self.build(environments: {}, outer_scope: {}, registers: {}, rethrow_errors: false, resource_limits: nil, static_environments: {})
|
def self.build(environments: {}, outer_scope: {}, registers: {}, rethrow_errors: false, resource_limits: nil, static_environments: {}, &block)
|
||||||
new(environments, outer_scope, registers, rethrow_errors, resource_limits, static_environments)
|
new(environments, outer_scope, registers, rethrow_errors, resource_limits, static_environments, &block)
|
||||||
end
|
end
|
||||||
|
|
||||||
def initialize(environments = {}, outer_scope = {}, registers = {}, rethrow_errors = false, resource_limits = nil, static_environments = {})
|
def initialize(environments = {}, outer_scope = {}, registers = {}, rethrow_errors = false, resource_limits = nil, static_environments = {})
|
||||||
@@ -34,16 +34,20 @@ module Liquid
|
|||||||
@strict_variables = false
|
@strict_variables = false
|
||||||
@resource_limits = resource_limits || ResourceLimits.new(Template.default_resource_limits)
|
@resource_limits = resource_limits || ResourceLimits.new(Template.default_resource_limits)
|
||||||
@base_scope_depth = 0
|
@base_scope_depth = 0
|
||||||
squash_instance_assigns_with_environments
|
|
||||||
|
|
||||||
self.exception_renderer = Template.default_exception_renderer
|
|
||||||
if rethrow_errors
|
|
||||||
self.exception_renderer = ->(_e) { raise }
|
|
||||||
end
|
|
||||||
|
|
||||||
@interrupts = []
|
@interrupts = []
|
||||||
@filters = []
|
@filters = []
|
||||||
@global_filter = nil
|
@global_filter = nil
|
||||||
|
@disabled_tags = {}
|
||||||
|
|
||||||
|
self.exception_renderer = Template.default_exception_renderer
|
||||||
|
if rethrow_errors
|
||||||
|
self.exception_renderer = Liquid::RAISE_EXCEPTION_LAMBDA
|
||||||
|
end
|
||||||
|
|
||||||
|
yield self if block_given?
|
||||||
|
|
||||||
|
# Do this last, since it could result in this object being passed to a Proc in the environment
|
||||||
|
squash_instance_assigns_with_environments
|
||||||
end
|
end
|
||||||
# rubocop:enable Metrics/ParameterLists
|
# rubocop:enable Metrics/ParameterLists
|
||||||
|
|
||||||
@@ -52,7 +56,7 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def strainer
|
def strainer
|
||||||
@strainer ||= Strainer.create(self, @filters)
|
@strainer ||= StrainerFactory.create(self, @filters)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Adds filters to this context.
|
# Adds filters to this context.
|
||||||
@@ -133,7 +137,7 @@ module Liquid
|
|||||||
def new_isolated_subcontext
|
def new_isolated_subcontext
|
||||||
check_overflow
|
check_overflow
|
||||||
|
|
||||||
Context.build(
|
self.class.build(
|
||||||
resource_limits: resource_limits,
|
resource_limits: resource_limits,
|
||||||
static_environments: static_environments,
|
static_environments: static_environments,
|
||||||
registers: StaticRegisters.new(registers)
|
registers: StaticRegisters.new(registers)
|
||||||
@@ -144,6 +148,7 @@ module Liquid
|
|||||||
subcontext.strainer = nil
|
subcontext.strainer = nil
|
||||||
subcontext.errors = errors
|
subcontext.errors = errors
|
||||||
subcontext.warnings = warnings
|
subcontext.warnings = warnings
|
||||||
|
subcontext.disabled_tags = @disabled_tags
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -208,9 +213,24 @@ module Liquid
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def with_disabled_tags(tag_names)
|
||||||
|
tag_names.each do |name|
|
||||||
|
@disabled_tags[name] = @disabled_tags.fetch(name, 0) + 1
|
||||||
|
end
|
||||||
|
yield
|
||||||
|
ensure
|
||||||
|
tag_names.each do |name|
|
||||||
|
@disabled_tags[name] -= 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def tag_disabled?(tag_name)
|
||||||
|
@disabled_tags.fetch(tag_name, 0) > 0
|
||||||
|
end
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
|
||||||
attr_writer :base_scope_depth, :warnings, :errors, :strainer, :filters
|
attr_writer :base_scope_depth, :warnings, :errors, :strainer, :filters, :disabled_tags
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
|||||||
@@ -1,23 +1,34 @@
|
|||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
class Document < BlockBody
|
class Document
|
||||||
def self.parse(tokens, parse_context)
|
def self.parse(tokens, parse_context)
|
||||||
doc = new
|
doc = new(parse_context)
|
||||||
doc.parse(tokens, parse_context)
|
doc.parse(tokens, parse_context)
|
||||||
doc
|
doc
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse(tokens, parse_context)
|
attr_reader :parse_context, :body
|
||||||
super do |end_tag_name, _end_tag_params|
|
|
||||||
unknown_tag(end_tag_name, parse_context) if end_tag_name
|
def initialize(parse_context)
|
||||||
|
@parse_context = parse_context
|
||||||
|
@body = new_body
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def nodelist
|
||||||
|
@body.nodelist
|
||||||
|
end
|
||||||
|
|
||||||
|
def parse(tokenizer, parse_context)
|
||||||
|
while parse_body(tokenizer)
|
||||||
|
end
|
||||||
|
@body.freeze
|
||||||
rescue SyntaxError => e
|
rescue SyntaxError => e
|
||||||
e.line_number ||= parse_context.line_number
|
e.line_number ||= parse_context.line_number
|
||||||
raise
|
raise
|
||||||
end
|
end
|
||||||
|
|
||||||
def unknown_tag(tag, parse_context)
|
def unknown_tag(tag, _markup, _tokenizer)
|
||||||
case tag
|
case tag
|
||||||
when 'else', 'end'
|
when 'else', 'end'
|
||||||
raise SyntaxError, parse_context.locale.t("errors.syntax.unexpected_outer_tag", tag: tag)
|
raise SyntaxError, parse_context.locale.t("errors.syntax.unexpected_outer_tag", tag: tag)
|
||||||
@@ -25,5 +36,30 @@ module Liquid
|
|||||||
raise SyntaxError, parse_context.locale.t("errors.syntax.unknown_tag", tag: tag)
|
raise SyntaxError, parse_context.locale.t("errors.syntax.unknown_tag", tag: tag)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def render_to_output_buffer(context, output)
|
||||||
|
@body.render_to_output_buffer(context, output)
|
||||||
|
end
|
||||||
|
|
||||||
|
def render(context)
|
||||||
|
render_to_output_buffer(context, +'')
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def new_body
|
||||||
|
parse_context.new_block_body
|
||||||
|
end
|
||||||
|
|
||||||
|
def parse_body(tokenizer)
|
||||||
|
@body.parse(tokenizer, parse_context) do |unknown_tag_name, unknown_tag_markup|
|
||||||
|
if unknown_tag_name
|
||||||
|
unknown_tag(unknown_tag_name, unknown_tag_markup, tokenizer)
|
||||||
|
true
|
||||||
|
else
|
||||||
|
false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -46,7 +46,6 @@ module Liquid
|
|||||||
StandardError = Class.new(Error)
|
StandardError = Class.new(Error)
|
||||||
SyntaxError = Class.new(Error)
|
SyntaxError = Class.new(Error)
|
||||||
StackLevelError = Class.new(Error)
|
StackLevelError = Class.new(Error)
|
||||||
TaintedError = Class.new(Error)
|
|
||||||
MemoryError = Class.new(Error)
|
MemoryError = Class.new(Error)
|
||||||
ZeroDivisionError = Class.new(Error)
|
ZeroDivisionError = Class.new(Error)
|
||||||
FloatDomainError = Class.new(Error)
|
FloatDomainError = Class.new(Error)
|
||||||
@@ -54,5 +53,6 @@ module Liquid
|
|||||||
UndefinedDropMethod = Class.new(Error)
|
UndefinedDropMethod = Class.new(Error)
|
||||||
UndefinedFilter = Class.new(Error)
|
UndefinedFilter = Class.new(Error)
|
||||||
MethodOverrideError = Class.new(Error)
|
MethodOverrideError = Class.new(Error)
|
||||||
|
DisabledError = Class.new(Error)
|
||||||
InternalError = Class.new(Error)
|
InternalError = Class.new(Error)
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -2,38 +2,27 @@
|
|||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
class Expression
|
class Expression
|
||||||
class MethodLiteral
|
|
||||||
attr_reader :method_name, :to_s
|
|
||||||
|
|
||||||
def initialize(method_name, to_s)
|
|
||||||
@method_name = method_name
|
|
||||||
@to_s = to_s
|
|
||||||
end
|
|
||||||
|
|
||||||
def to_liquid
|
|
||||||
to_s
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
LITERALS = {
|
LITERALS = {
|
||||||
nil => nil, 'nil' => nil, 'null' => nil, '' => nil,
|
nil => nil, 'nil' => nil, 'null' => nil, '' => nil,
|
||||||
'true' => true,
|
'true' => true,
|
||||||
'false' => false,
|
'false' => false,
|
||||||
'blank' => MethodLiteral.new(:blank?, '').freeze,
|
'blank' => '',
|
||||||
'empty' => MethodLiteral.new(:empty?, '').freeze
|
'empty' => ''
|
||||||
}.freeze
|
}.freeze
|
||||||
|
|
||||||
SINGLE_QUOTED_STRING = /\A'(.*)'\z/m
|
SINGLE_QUOTED_STRING = /\A\s*'(.*)'\s*\z/m
|
||||||
DOUBLE_QUOTED_STRING = /\A"(.*)"\z/m
|
DOUBLE_QUOTED_STRING = /\A\s*"(.*)"\s*\z/m
|
||||||
INTEGERS_REGEX = /\A(-?\d+)\z/
|
INTEGERS_REGEX = /\A\s*(-?\d+)\s*\z/
|
||||||
FLOATS_REGEX = /\A(-?\d[\d\.]+)\z/
|
FLOATS_REGEX = /\A\s*(-?\d[\d\.]+)\s*\z/
|
||||||
RANGES_REGEX = /\A\((\S+)\.\.(\S+)\)\z/
|
|
||||||
|
# Use an atomic group (?>...) to avoid pathological backtracing from
|
||||||
|
# malicious input as described in https://github.com/Shopify/liquid/issues/1357
|
||||||
|
RANGES_REGEX = /\A\s*\(\s*(?>(\S+)\s*\.\.)\s*(\S+)\s*\)\s*\z/
|
||||||
|
|
||||||
def self.parse(markup)
|
def self.parse(markup)
|
||||||
if LITERALS.key?(markup)
|
|
||||||
LITERALS[markup]
|
|
||||||
else
|
|
||||||
case markup
|
case markup
|
||||||
|
when nil
|
||||||
|
nil
|
||||||
when SINGLE_QUOTED_STRING, DOUBLE_QUOTED_STRING
|
when SINGLE_QUOTED_STRING, DOUBLE_QUOTED_STRING
|
||||||
Regexp.last_match(1)
|
Regexp.last_match(1)
|
||||||
when INTEGERS_REGEX
|
when INTEGERS_REGEX
|
||||||
@@ -42,6 +31,10 @@ module Liquid
|
|||||||
RangeLookup.parse(Regexp.last_match(1), Regexp.last_match(2))
|
RangeLookup.parse(Regexp.last_match(1), Regexp.last_match(2))
|
||||||
when FLOATS_REGEX
|
when FLOATS_REGEX
|
||||||
Regexp.last_match(1).to_f
|
Regexp.last_match(1).to_f
|
||||||
|
else
|
||||||
|
markup = markup.strip
|
||||||
|
if LITERALS.key?(markup)
|
||||||
|
LITERALS[markup]
|
||||||
else
|
else
|
||||||
VariableLookup.parse(markup)
|
VariableLookup.parse(markup)
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -9,7 +9,12 @@ module Liquid
|
|||||||
@index = 0
|
@index = 0
|
||||||
end
|
end
|
||||||
|
|
||||||
attr_reader :name, :length, :parentloop
|
attr_reader :length, :parentloop
|
||||||
|
|
||||||
|
def name
|
||||||
|
Usage.increment('forloop_drop_name')
|
||||||
|
@name
|
||||||
|
end
|
||||||
|
|
||||||
def index
|
def index
|
||||||
@index + 1
|
@index + 1
|
||||||
|
|||||||
@@ -20,9 +20,9 @@
|
|||||||
tag_termination: "Tag '%{token}' was not properly terminated with regexp: %{tag_end}"
|
tag_termination: "Tag '%{token}' was not properly terminated with regexp: %{tag_end}"
|
||||||
variable_termination: "Variable '%{token}' was not properly terminated with regexp: %{tag_end}"
|
variable_termination: "Variable '%{token}' was not properly terminated with regexp: %{tag_end}"
|
||||||
tag_never_closed: "'%{block_name}' tag was never closed"
|
tag_never_closed: "'%{block_name}' tag was never closed"
|
||||||
meta_syntax_error: "Liquid syntax error: #{e.message}"
|
|
||||||
table_row: "Syntax Error in 'table_row loop' - Valid syntax: table_row [item] in [collection] cols=3"
|
table_row: "Syntax Error in 'table_row loop' - Valid syntax: table_row [item] in [collection] cols=3"
|
||||||
render: "Syntax error in tag 'render' - Template name must be a quoted string"
|
render: "Syntax error in tag 'render' - Template name must be a quoted string"
|
||||||
|
inline_comment_invalid: "Syntax error in tag '#' - Each line of comments must be prefixed by the '#' character"
|
||||||
argument:
|
argument:
|
||||||
include: "Argument error in tag 'include' - Illegal template name"
|
include: "Argument error in tag 'include' - Illegal template name"
|
||||||
disabled:
|
disabled:
|
||||||
|
|||||||
@@ -7,8 +7,10 @@ module Liquid
|
|||||||
|
|
||||||
def initialize(options = {})
|
def initialize(options = {})
|
||||||
@template_options = options ? options.dup : {}
|
@template_options = options ? options.dup : {}
|
||||||
|
|
||||||
@locale = @template_options[:locale] ||= I18n.new
|
@locale = @template_options[:locale] ||= I18n.new
|
||||||
@warnings = []
|
@warnings = []
|
||||||
|
|
||||||
self.depth = 0
|
self.depth = 0
|
||||||
self.partial = false
|
self.partial = false
|
||||||
end
|
end
|
||||||
@@ -17,9 +19,22 @@ module Liquid
|
|||||||
@options[option_key]
|
@options[option_key]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def new_block_body
|
||||||
|
Liquid::BlockBody.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def new_tokenizer(markup, start_line_number: nil, for_liquid_tag: false)
|
||||||
|
Tokenizer.new(markup, line_number: start_line_number, for_liquid_tag: for_liquid_tag)
|
||||||
|
end
|
||||||
|
|
||||||
|
def parse_expression(markup)
|
||||||
|
Expression.parse(markup)
|
||||||
|
end
|
||||||
|
|
||||||
def partial=(value)
|
def partial=(value)
|
||||||
@partial = value
|
@partial = value
|
||||||
@options = value ? partial_options : @template_options
|
@options = value ? partial_options : @template_options
|
||||||
|
|
||||||
@error_mode = @options[:error_mode] || Template.error_mode
|
@error_mode = @options[:error_mode] || Template.error_mode
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -46,16 +46,20 @@ module Liquid
|
|||||||
tok[0] == type
|
tok[0] == type
|
||||||
end
|
end
|
||||||
|
|
||||||
SINGLE_TOKEN_EXPRESSION_TYPES = [:string, :number].freeze
|
|
||||||
private_constant :SINGLE_TOKEN_EXPRESSION_TYPES
|
|
||||||
|
|
||||||
def expression
|
def expression
|
||||||
token = @tokens[@p]
|
token = @tokens[@p]
|
||||||
if token[0] == :id
|
case token[0]
|
||||||
variable_signature
|
when :id
|
||||||
elsif SINGLE_TOKEN_EXPRESSION_TYPES.include?(token[0])
|
str = consume
|
||||||
|
str << variable_lookups
|
||||||
|
when :open_square
|
||||||
|
str = consume
|
||||||
|
str << expression
|
||||||
|
str << consume(:close_square)
|
||||||
|
str << variable_lookups
|
||||||
|
when :string, :number
|
||||||
consume
|
consume
|
||||||
elsif token.first == :open_round
|
when :open_round
|
||||||
consume
|
consume
|
||||||
first = expression
|
first = expression
|
||||||
consume(:dotdot)
|
consume(:dotdot)
|
||||||
@@ -78,16 +82,19 @@ module Liquid
|
|||||||
str
|
str
|
||||||
end
|
end
|
||||||
|
|
||||||
def variable_signature
|
def variable_lookups
|
||||||
str = consume(:id)
|
str = +""
|
||||||
while look(:open_square)
|
loop do
|
||||||
|
if look(:open_square)
|
||||||
str << consume
|
str << consume
|
||||||
str << expression
|
str << expression
|
||||||
str << consume(:close_square)
|
str << consume(:close_square)
|
||||||
end
|
elsif look(:dot)
|
||||||
if look(:dot)
|
|
||||||
str << consume
|
str << consume
|
||||||
str << variable_signature
|
str << consume(:id)
|
||||||
|
else
|
||||||
|
break
|
||||||
|
end
|
||||||
end
|
end
|
||||||
str
|
str
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -2,16 +2,28 @@
|
|||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
module ParserSwitching
|
module ParserSwitching
|
||||||
|
def strict_parse_with_error_mode_fallback(markup)
|
||||||
|
strict_parse_with_error_context(markup)
|
||||||
|
rescue SyntaxError => e
|
||||||
|
case parse_context.error_mode
|
||||||
|
when :strict
|
||||||
|
raise
|
||||||
|
when :warn
|
||||||
|
parse_context.warnings << e
|
||||||
|
end
|
||||||
|
lax_parse(markup)
|
||||||
|
end
|
||||||
|
|
||||||
def parse_with_selected_parser(markup)
|
def parse_with_selected_parser(markup)
|
||||||
case parse_context.error_mode
|
case parse_context.error_mode
|
||||||
when :strict then strict_parse_with_error_context(markup)
|
when :strict then strict_parse_with_error_context(markup)
|
||||||
when :lax then lax_parse(markup)
|
when :lax then lax_parse(markup)
|
||||||
when :warn
|
when :warn
|
||||||
begin
|
begin
|
||||||
return strict_parse_with_error_context(markup)
|
strict_parse_with_error_context(markup)
|
||||||
rescue SyntaxError => e
|
rescue SyntaxError => e
|
||||||
parse_context.warnings << e
|
parse_context.warnings << e
|
||||||
return lax_parse(markup)
|
lax_parse(markup)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -9,9 +9,13 @@ module Liquid
|
|||||||
|
|
||||||
file_system = (context.registers[:file_system] ||= Liquid::Template.file_system)
|
file_system = (context.registers[:file_system] ||= Liquid::Template.file_system)
|
||||||
source = file_system.read_template_file(template_name)
|
source = file_system.read_template_file(template_name)
|
||||||
|
|
||||||
parse_context.partial = true
|
parse_context.partial = true
|
||||||
|
|
||||||
partial = Liquid::Template.parse(source, parse_context)
|
template_factory = (context.registers[:template_factory] ||= Liquid::TemplateFactory.new)
|
||||||
|
template = template_factory.for(template_name)
|
||||||
|
|
||||||
|
partial = template.parse(source, parse_context)
|
||||||
cached_partials[template_name] = partial
|
cached_partials[template_name] = partial
|
||||||
ensure
|
ensure
|
||||||
parse_context.partial = false
|
parse_context.partial = false
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ module Liquid
|
|||||||
# node.code
|
# node.code
|
||||||
#
|
#
|
||||||
# # Which template and line number of this node.
|
# # Which template and line number of this node.
|
||||||
# # If top level, this will be "<root>".
|
# # The top-level template name is `nil` by default, but can be set in the Liquid::Context before rendering.
|
||||||
# node.partial
|
# node.partial
|
||||||
# node.line_number
|
# node.line_number
|
||||||
#
|
#
|
||||||
@@ -46,115 +46,94 @@ module Liquid
|
|||||||
include Enumerable
|
include Enumerable
|
||||||
|
|
||||||
class Timing
|
class Timing
|
||||||
attr_reader :code, :partial, :line_number, :children
|
attr_reader :code, :template_name, :line_number, :children
|
||||||
|
attr_accessor :total_time
|
||||||
|
alias_method :render_time, :total_time
|
||||||
|
alias_method :partial, :template_name
|
||||||
|
|
||||||
def initialize(node, partial)
|
def initialize(code: nil, template_name: nil, line_number: nil)
|
||||||
@code = node.respond_to?(:raw) ? node.raw : node
|
@code = code
|
||||||
@partial = partial
|
@template_name = template_name
|
||||||
@line_number = node.respond_to?(:line_number) ? node.line_number : nil
|
@line_number = line_number
|
||||||
@children = []
|
@children = []
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.start(node, partial)
|
def self_time
|
||||||
new(node, partial).tap(&:start)
|
@self_time ||= begin
|
||||||
|
total_children_time = 0.0
|
||||||
|
@children.each do |child|
|
||||||
|
total_children_time += child.total_time
|
||||||
end
|
end
|
||||||
|
@total_time - total_children_time
|
||||||
def start
|
|
||||||
@start_time = Time.now
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def finish
|
|
||||||
@end_time = Time.now
|
|
||||||
end
|
|
||||||
|
|
||||||
def render_time
|
|
||||||
@end_time - @start_time
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.profile_node_render(node)
|
attr_reader :total_time
|
||||||
if Profiler.current_profile && node.respond_to?(:render)
|
alias_method :total_render_time, :total_time
|
||||||
Profiler.current_profile.start_node(node)
|
|
||||||
output = yield
|
|
||||||
Profiler.current_profile.end_node(node)
|
|
||||||
output
|
|
||||||
else
|
|
||||||
yield
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.profile_children(template_name)
|
|
||||||
if Profiler.current_profile
|
|
||||||
Profiler.current_profile.push_partial(template_name)
|
|
||||||
output = yield
|
|
||||||
Profiler.current_profile.pop_partial
|
|
||||||
output
|
|
||||||
else
|
|
||||||
yield
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.current_profile
|
|
||||||
Thread.current[:liquid_profiler]
|
|
||||||
end
|
|
||||||
|
|
||||||
def initialize
|
def initialize
|
||||||
@partial_stack = ["<root>"]
|
@root_children = []
|
||||||
|
@current_children = nil
|
||||||
@root_timing = Timing.new("", current_partial)
|
@total_time = 0.0
|
||||||
@timing_stack = [@root_timing]
|
|
||||||
|
|
||||||
@render_start_at = Time.now
|
|
||||||
@render_end_at = @render_start_at
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def start
|
def profile(template_name, &block)
|
||||||
Thread.current[:liquid_profiler] = self
|
# nested renders are done from a tag that already has a timing node
|
||||||
@render_start_at = Time.now
|
return yield if @current_children
|
||||||
|
|
||||||
|
root_children = @root_children
|
||||||
|
render_idx = root_children.length
|
||||||
|
begin
|
||||||
|
@current_children = root_children
|
||||||
|
profile_node(template_name, &block)
|
||||||
|
ensure
|
||||||
|
@current_children = nil
|
||||||
|
if (timing = root_children[render_idx])
|
||||||
|
@total_time += timing.total_time
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def stop
|
def children
|
||||||
Thread.current[:liquid_profiler] = nil
|
children = @root_children
|
||||||
@render_end_at = Time.now
|
if children.length == 1
|
||||||
|
children.first.children
|
||||||
|
else
|
||||||
|
children
|
||||||
end
|
end
|
||||||
|
|
||||||
def total_render_time
|
|
||||||
@render_end_at - @render_start_at
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def each(&block)
|
def each(&block)
|
||||||
@root_timing.children.each(&block)
|
children.each(&block)
|
||||||
end
|
end
|
||||||
|
|
||||||
def [](idx)
|
def [](idx)
|
||||||
@root_timing.children[idx]
|
children[idx]
|
||||||
end
|
end
|
||||||
|
|
||||||
def length
|
def length
|
||||||
@root_timing.children.length
|
children.length
|
||||||
end
|
end
|
||||||
|
|
||||||
def start_node(node)
|
def profile_node(template_name, code: nil, line_number: nil)
|
||||||
@timing_stack.push(Timing.start(node, current_partial))
|
timing = Timing.new(code: code, template_name: template_name, line_number: line_number)
|
||||||
|
parent_children = @current_children
|
||||||
|
start_time = monotonic_time
|
||||||
|
begin
|
||||||
|
@current_children = timing.children
|
||||||
|
yield
|
||||||
|
ensure
|
||||||
|
@current_children = parent_children
|
||||||
|
timing.total_time = monotonic_time - start_time
|
||||||
|
parent_children << timing
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def end_node(_node)
|
private
|
||||||
timing = @timing_stack.pop
|
|
||||||
timing.finish
|
|
||||||
|
|
||||||
@timing_stack.last.children << timing
|
def monotonic_time
|
||||||
end
|
Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
||||||
|
|
||||||
def current_partial
|
|
||||||
@partial_stack.last
|
|
||||||
end
|
|
||||||
|
|
||||||
def push_partial(partial_name)
|
|
||||||
@partial_stack.push(partial_name)
|
|
||||||
end
|
|
||||||
|
|
||||||
def pop_partial
|
|
||||||
@partial_stack.pop
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,25 +1,35 @@
|
|||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
class BlockBody
|
module BlockBodyProfilingHook
|
||||||
def render_node_with_profiling(context, output, node)
|
def render_node(context, output, node)
|
||||||
Profiler.profile_node_render(node) do
|
if (profiler = context.profiler)
|
||||||
render_node_without_profiling(context, output, node)
|
profiler.profile_node(context.template_name, code: node.raw, line_number: node.line_number) do
|
||||||
|
super
|
||||||
|
end
|
||||||
|
else
|
||||||
|
super
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
BlockBody.prepend(BlockBodyProfilingHook)
|
||||||
|
|
||||||
alias_method :render_node_without_profiling, :render_node
|
module DocumentProfilingHook
|
||||||
alias_method :render_node, :render_node_with_profiling
|
def render_to_output_buffer(context, output)
|
||||||
|
return super unless context.profiler
|
||||||
|
context.profiler.profile(context.template_name) { super }
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
Document.prepend(DocumentProfilingHook)
|
||||||
|
|
||||||
class Include < Tag
|
module ContextProfilingHook
|
||||||
def render_to_output_buffer_with_profiling(context, output)
|
attr_accessor :profiler
|
||||||
Profiler.profile_children(context.evaluate(@template_name_expr).to_s) do
|
|
||||||
render_to_output_buffer_without_profiling(context, output)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
alias_method :render_to_output_buffer_without_profiling, :render_to_output_buffer
|
def new_isolated_subcontext
|
||||||
alias_method :render_to_output_buffer, :render_to_output_buffer_with_profiling
|
new_context = super
|
||||||
|
new_context.profiler = profiler
|
||||||
|
new_context
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
Context.prepend(ContextProfilingHook)
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,32 +0,0 @@
|
|||||||
# frozen_string_literal: true
|
|
||||||
module Liquid
|
|
||||||
class DisabledTags < Register
|
|
||||||
def initialize
|
|
||||||
@disabled_tags = {}
|
|
||||||
end
|
|
||||||
|
|
||||||
def disabled?(tag)
|
|
||||||
@disabled_tags.key?(tag) && @disabled_tags[tag] > 0
|
|
||||||
end
|
|
||||||
|
|
||||||
def disable(tags)
|
|
||||||
tags.each(&method(:increment))
|
|
||||||
yield
|
|
||||||
ensure
|
|
||||||
tags.each(&method(:decrement))
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def increment(tag)
|
|
||||||
@disabled_tags[tag] ||= 0
|
|
||||||
@disabled_tags[tag] += 1
|
|
||||||
end
|
|
||||||
|
|
||||||
def decrement(tag)
|
|
||||||
@disabled_tags[tag] -= 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
Template.add_register(:disabled_tags, DisabledTags.new)
|
|
||||||
end
|
|
||||||
@@ -2,8 +2,8 @@
|
|||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
class ResourceLimits
|
class ResourceLimits
|
||||||
attr_accessor :render_length, :render_score, :assign_score,
|
attr_accessor :render_length_limit, :render_score_limit, :assign_score_limit
|
||||||
:render_length_limit, :render_score_limit, :assign_score_limit
|
attr_reader :render_score, :assign_score
|
||||||
|
|
||||||
def initialize(limits)
|
def initialize(limits)
|
||||||
@render_length_limit = limits[:render_length_limit]
|
@render_length_limit = limits[:render_length_limit]
|
||||||
@@ -12,14 +12,51 @@ module Liquid
|
|||||||
reset
|
reset
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def increment_render_score(amount)
|
||||||
|
@render_score += amount
|
||||||
|
raise_limits_reached if @render_score_limit && @render_score > @render_score_limit
|
||||||
|
end
|
||||||
|
|
||||||
|
def increment_assign_score(amount)
|
||||||
|
@assign_score += amount
|
||||||
|
raise_limits_reached if @assign_score_limit && @assign_score > @assign_score_limit
|
||||||
|
end
|
||||||
|
|
||||||
|
# update either render_length or assign_score based on whether or not the writes are captured
|
||||||
|
def increment_write_score(output)
|
||||||
|
if (last_captured = @last_capture_length)
|
||||||
|
captured = output.bytesize
|
||||||
|
increment = captured - last_captured
|
||||||
|
@last_capture_length = captured
|
||||||
|
increment_assign_score(increment)
|
||||||
|
elsif @render_length_limit && output.bytesize > @render_length_limit
|
||||||
|
raise_limits_reached
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def raise_limits_reached
|
||||||
|
@reached_limit = true
|
||||||
|
raise MemoryError, "Memory limits exceeded"
|
||||||
|
end
|
||||||
|
|
||||||
def reached?
|
def reached?
|
||||||
(@render_length_limit && @render_length > @render_length_limit) ||
|
@reached_limit
|
||||||
(@render_score_limit && @render_score > @render_score_limit) ||
|
|
||||||
(@assign_score_limit && @assign_score > @assign_score_limit)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def reset
|
def reset
|
||||||
@render_length = @render_score = @assign_score = 0
|
@reached_limit = false
|
||||||
|
@last_capture_length = nil
|
||||||
|
@render_score = @assign_score = 0
|
||||||
|
end
|
||||||
|
|
||||||
|
def with_capture
|
||||||
|
old_capture_length = @last_capture_length
|
||||||
|
begin
|
||||||
|
@last_capture_length = 0
|
||||||
|
yield
|
||||||
|
ensure
|
||||||
|
@last_capture_length = old_capture_length
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def escape(input)
|
def escape(input)
|
||||||
CGI.escapeHTML(input.to_s).untaint unless input.nil?
|
CGI.escapeHTML(input.to_s) unless input.nil?
|
||||||
end
|
end
|
||||||
alias_method :h, :escape
|
alias_method :h, :escape
|
||||||
|
|
||||||
@@ -78,19 +78,26 @@ module Liquid
|
|||||||
return if input.nil?
|
return if input.nil?
|
||||||
input_str = input.to_s
|
input_str = input.to_s
|
||||||
length = Utils.to_integer(length)
|
length = Utils.to_integer(length)
|
||||||
|
|
||||||
truncate_string_str = truncate_string.to_s
|
truncate_string_str = truncate_string.to_s
|
||||||
|
|
||||||
l = length - truncate_string_str.length
|
l = length - truncate_string_str.length
|
||||||
l = 0 if l < 0
|
l = 0 if l < 0
|
||||||
|
|
||||||
input_str.length > length ? input_str[0...l].concat(truncate_string_str) : input_str
|
input_str.length > length ? input_str[0...l].concat(truncate_string_str) : input_str
|
||||||
end
|
end
|
||||||
|
|
||||||
def truncatewords(input, words = 15, truncate_string = "...")
|
def truncatewords(input, words = 15, truncate_string = "...")
|
||||||
return if input.nil?
|
return if input.nil?
|
||||||
wordlist = input.to_s.split
|
input = input.to_s
|
||||||
words = Utils.to_integer(words)
|
words = Utils.to_integer(words)
|
||||||
l = words - 1
|
words = 1 if words <= 0
|
||||||
l = 0 if l < 0
|
|
||||||
wordlist.length > l ? wordlist[0..l].join(" ").concat(truncate_string.to_s) : input
|
wordlist = input.split(" ", words + 1)
|
||||||
|
return input if wordlist.length <= words
|
||||||
|
|
||||||
|
wordlist.pop
|
||||||
|
wordlist.join(" ").concat(truncate_string.to_s)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Split input string into an array of substrings separated by given pattern.
|
# Split input string into an array of substrings separated by given pattern.
|
||||||
@@ -128,13 +135,13 @@ module Liquid
|
|||||||
|
|
||||||
# Join elements of the array with certain character between them
|
# Join elements of the array with certain character between them
|
||||||
def join(input, glue = ' ')
|
def join(input, glue = ' ')
|
||||||
InputIterator.new(input).join(glue)
|
InputIterator.new(input, context).join(glue)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Sort elements of the array
|
# Sort elements of the array
|
||||||
# provide optional property with which to sort an array of hashes or drops
|
# provide optional property with which to sort an array of hashes or drops
|
||||||
def sort(input, property = nil)
|
def sort(input, property = nil)
|
||||||
ary = InputIterator.new(input)
|
ary = InputIterator.new(input, context)
|
||||||
|
|
||||||
return [] if ary.empty?
|
return [] if ary.empty?
|
||||||
|
|
||||||
@@ -154,7 +161,7 @@ module Liquid
|
|||||||
# Sort elements of an array ignoring case if strings
|
# Sort elements of an array ignoring case if strings
|
||||||
# provide optional property with which to sort an array of hashes or drops
|
# provide optional property with which to sort an array of hashes or drops
|
||||||
def sort_natural(input, property = nil)
|
def sort_natural(input, property = nil)
|
||||||
ary = InputIterator.new(input)
|
ary = InputIterator.new(input, context)
|
||||||
|
|
||||||
return [] if ary.empty?
|
return [] if ary.empty?
|
||||||
|
|
||||||
@@ -174,7 +181,7 @@ module Liquid
|
|||||||
# Filter the elements of an array to those with a certain property value.
|
# Filter the elements of an array to those with a certain property value.
|
||||||
# By default the target is any truthy value.
|
# By default the target is any truthy value.
|
||||||
def where(input, property, target_value = nil)
|
def where(input, property, target_value = nil)
|
||||||
ary = InputIterator.new(input)
|
ary = InputIterator.new(input, context)
|
||||||
|
|
||||||
if ary.empty?
|
if ary.empty?
|
||||||
[]
|
[]
|
||||||
@@ -196,7 +203,7 @@ module Liquid
|
|||||||
# Remove duplicate elements from an array
|
# Remove duplicate elements from an array
|
||||||
# provide optional property with which to determine uniqueness
|
# provide optional property with which to determine uniqueness
|
||||||
def uniq(input, property = nil)
|
def uniq(input, property = nil)
|
||||||
ary = InputIterator.new(input)
|
ary = InputIterator.new(input, context)
|
||||||
|
|
||||||
if property.nil?
|
if property.nil?
|
||||||
ary.uniq
|
ary.uniq
|
||||||
@@ -213,13 +220,13 @@ module Liquid
|
|||||||
|
|
||||||
# Reverse the elements of an array
|
# Reverse the elements of an array
|
||||||
def reverse(input)
|
def reverse(input)
|
||||||
ary = InputIterator.new(input)
|
ary = InputIterator.new(input, context)
|
||||||
ary.reverse
|
ary.reverse
|
||||||
end
|
end
|
||||||
|
|
||||||
# map/collect on a given property
|
# map/collect on a given property
|
||||||
def map(input, property)
|
def map(input, property)
|
||||||
InputIterator.new(input).map do |e|
|
InputIterator.new(input, context).map do |e|
|
||||||
e = e.call if e.is_a?(Proc)
|
e = e.call if e.is_a?(Proc)
|
||||||
|
|
||||||
if property == "to_liquid"
|
if property == "to_liquid"
|
||||||
@@ -236,7 +243,7 @@ module Liquid
|
|||||||
# Remove nils within an array
|
# Remove nils within an array
|
||||||
# provide optional property with which to check for nil
|
# provide optional property with which to check for nil
|
||||||
def compact(input, property = nil)
|
def compact(input, property = nil)
|
||||||
ary = InputIterator.new(input)
|
ary = InputIterator.new(input, context)
|
||||||
|
|
||||||
if property.nil?
|
if property.nil?
|
||||||
ary.compact
|
ary.compact
|
||||||
@@ -280,7 +287,7 @@ module Liquid
|
|||||||
unless array.respond_to?(:to_ary)
|
unless array.respond_to?(:to_ary)
|
||||||
raise ArgumentError, "concat filter requires an array argument"
|
raise ArgumentError, "concat filter requires an array argument"
|
||||||
end
|
end
|
||||||
InputIterator.new(input).concat(array)
|
InputIterator.new(input, context).concat(array)
|
||||||
end
|
end
|
||||||
|
|
||||||
# prepend a string to another
|
# prepend a string to another
|
||||||
@@ -290,7 +297,7 @@ module Liquid
|
|||||||
|
|
||||||
# Add <br /> tags in front of all newlines in input string
|
# Add <br /> tags in front of all newlines in input string
|
||||||
def newline_to_br(input)
|
def newline_to_br(input)
|
||||||
input.to_s.gsub(/\n/, "<br />\n")
|
input.to_s.gsub(/\r?\n/, "<br />\n")
|
||||||
end
|
end
|
||||||
|
|
||||||
# Reformat a date using Ruby's core Time#strftime( string ) -> string
|
# Reformat a date using Ruby's core Time#strftime( string ) -> string
|
||||||
@@ -421,17 +428,26 @@ module Liquid
|
|||||||
result.is_a?(BigDecimal) ? result.to_f : result
|
result.is_a?(BigDecimal) ? result.to_f : result
|
||||||
end
|
end
|
||||||
|
|
||||||
def default(input, default_value = '')
|
# Set a default value when the input is nil, false or empty
|
||||||
if !input || input.respond_to?(:empty?) && input.empty?
|
#
|
||||||
Usage.increment("default_filter_received_false_value") if input == false # See https://github.com/Shopify/liquid/issues/1127
|
# Example:
|
||||||
default_value
|
# {{ product.title | default: "No Title" }}
|
||||||
else
|
#
|
||||||
input
|
# Use `allow_false` when an input should only be tested against nil or empty and not false.
|
||||||
end
|
#
|
||||||
|
# Example:
|
||||||
|
# {{ product.title | default: "No Title", allow_false: true }}
|
||||||
|
#
|
||||||
|
def default(input, default_value = '', options = {})
|
||||||
|
options = {} unless options.is_a?(Hash)
|
||||||
|
false_check = options['allow_false'] ? input.nil? : !input
|
||||||
|
false_check || (input.respond_to?(:empty?) && input.empty?) ? default_value : input
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
attr_reader :context
|
||||||
|
|
||||||
def raise_property_error(property)
|
def raise_property_error(property)
|
||||||
raise Liquid::ArgumentError, "cannot select the property '#{property}'"
|
raise Liquid::ArgumentError, "cannot select the property '#{property}'"
|
||||||
end
|
end
|
||||||
@@ -460,7 +476,8 @@ module Liquid
|
|||||||
class InputIterator
|
class InputIterator
|
||||||
include Enumerable
|
include Enumerable
|
||||||
|
|
||||||
def initialize(input)
|
def initialize(input, context)
|
||||||
|
@context = context
|
||||||
@input = if input.is_a?(Array)
|
@input = if input.is_a?(Array)
|
||||||
input.flatten
|
input.flatten
|
||||||
elsif input.is_a?(Hash)
|
elsif input.is_a?(Hash)
|
||||||
@@ -499,6 +516,7 @@ module Liquid
|
|||||||
|
|
||||||
def each
|
def each
|
||||||
@input.each do |e|
|
@input.each do |e|
|
||||||
|
e.context = @context if e.respond_to?(:context=)
|
||||||
yield(e.respond_to?(:to_liquid) ? e.to_liquid : e)
|
yield(e.respond_to?(:to_liquid) ? e.to_liquid : e)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
class StaticRegisters
|
class StaticRegisters
|
||||||
attr_reader :static, :registers
|
attr_reader :static
|
||||||
|
|
||||||
def initialize(registers = {})
|
def initialize(registers = {})
|
||||||
@static = registers.is_a?(StaticRegisters) ? registers.static : registers
|
@static = registers.is_a?(StaticRegisters) ? registers.static : registers
|
||||||
@@ -25,8 +25,16 @@ module Liquid
|
|||||||
@registers.delete(key)
|
@registers.delete(key)
|
||||||
end
|
end
|
||||||
|
|
||||||
def fetch(key, default = nil)
|
UNDEFINED = Object.new
|
||||||
key?(key) ? self[key] : default
|
|
||||||
|
def fetch(key, default = UNDEFINED, &block)
|
||||||
|
if @registers.key?(key)
|
||||||
|
@registers.fetch(key)
|
||||||
|
elsif default != UNDEFINED
|
||||||
|
@static.fetch(key, default, &block)
|
||||||
|
else
|
||||||
|
@static.fetch(key, &block)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def key?(key)
|
def key?(key)
|
||||||
|
|||||||
@@ -1,68 +0,0 @@
|
|||||||
# frozen_string_literal: true
|
|
||||||
|
|
||||||
require 'set'
|
|
||||||
|
|
||||||
module Liquid
|
|
||||||
# Strainer is the parent class for the filters system.
|
|
||||||
# New filters are mixed into the strainer class which is then instantiated for each liquid template render run.
|
|
||||||
#
|
|
||||||
# The Strainer only allows method calls defined in filters given to it via Strainer.global_filter,
|
|
||||||
# Context#add_filters or Template.register_filter
|
|
||||||
class Strainer #:nodoc:
|
|
||||||
@@global_strainer = Class.new(Strainer) do
|
|
||||||
@filter_methods = Set.new
|
|
||||||
end
|
|
||||||
@@strainer_class_cache = Hash.new do |hash, filters|
|
|
||||||
hash[filters] = Class.new(@@global_strainer) do
|
|
||||||
@filter_methods = @@global_strainer.filter_methods.dup
|
|
||||||
filters.each { |f| add_filter(f) }
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def initialize(context)
|
|
||||||
@context = context
|
|
||||||
end
|
|
||||||
|
|
||||||
class << self
|
|
||||||
attr_reader :filter_methods
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.add_filter(filter)
|
|
||||||
raise ArgumentError, "Expected module but got: #{filter.class}" unless filter.is_a?(Module)
|
|
||||||
unless include?(filter)
|
|
||||||
invokable_non_public_methods = (filter.private_instance_methods + filter.protected_instance_methods).select { |m| invokable?(m) }
|
|
||||||
if invokable_non_public_methods.any?
|
|
||||||
raise MethodOverrideError, "Filter overrides registered public methods as non public: #{invokable_non_public_methods.join(', ')}"
|
|
||||||
else
|
|
||||||
send(:include, filter)
|
|
||||||
@filter_methods.merge(filter.public_instance_methods.map(&:to_s))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.global_filter(filter)
|
|
||||||
@@strainer_class_cache.clear
|
|
||||||
@@global_strainer.add_filter(filter)
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.invokable?(method)
|
|
||||||
@filter_methods.include?(method.to_s)
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.create(context, filters = [])
|
|
||||||
@@strainer_class_cache[filters].new(context)
|
|
||||||
end
|
|
||||||
|
|
||||||
def invoke(method, *args)
|
|
||||||
if self.class.invokable?(method)
|
|
||||||
send(method, *args)
|
|
||||||
elsif @context&.strict_filters
|
|
||||||
raise Liquid::UndefinedFilter, "undefined filter #{method}"
|
|
||||||
else
|
|
||||||
args.first
|
|
||||||
end
|
|
||||||
rescue ::ArgumentError => e
|
|
||||||
raise Liquid::ArgumentError, e.message, e.backtrace
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
36
lib/liquid/strainer_factory.rb
Normal file
36
lib/liquid/strainer_factory.rb
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Liquid
|
||||||
|
# StrainerFactory is the factory for the filters system.
|
||||||
|
module StrainerFactory
|
||||||
|
extend self
|
||||||
|
|
||||||
|
def add_global_filter(filter)
|
||||||
|
strainer_class_cache.clear
|
||||||
|
global_filters << filter
|
||||||
|
end
|
||||||
|
|
||||||
|
def create(context, filters = [])
|
||||||
|
strainer_from_cache(filters).new(context)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def global_filters
|
||||||
|
@global_filters ||= []
|
||||||
|
end
|
||||||
|
|
||||||
|
def strainer_from_cache(filters)
|
||||||
|
strainer_class_cache[filters] ||= begin
|
||||||
|
klass = Class.new(StrainerTemplate)
|
||||||
|
global_filters.each { |f| klass.add_filter(f) }
|
||||||
|
filters.each { |f| klass.add_filter(f) }
|
||||||
|
klass
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def strainer_class_cache
|
||||||
|
@strainer_class_cache ||= {}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
53
lib/liquid/strainer_template.rb
Normal file
53
lib/liquid/strainer_template.rb
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'set'
|
||||||
|
|
||||||
|
module Liquid
|
||||||
|
# StrainerTemplate is the computed class for the filters system.
|
||||||
|
# New filters are mixed into the strainer class which is then instantiated for each liquid template render run.
|
||||||
|
#
|
||||||
|
# The Strainer only allows method calls defined in filters given to it via StrainerFactory.add_global_filter,
|
||||||
|
# Context#add_filters or Template.register_filter
|
||||||
|
class StrainerTemplate
|
||||||
|
def initialize(context)
|
||||||
|
@context = context
|
||||||
|
end
|
||||||
|
|
||||||
|
class << self
|
||||||
|
def add_filter(filter)
|
||||||
|
return if include?(filter)
|
||||||
|
|
||||||
|
invokable_non_public_methods = (filter.private_instance_methods + filter.protected_instance_methods).select { |m| invokable?(m) }
|
||||||
|
if invokable_non_public_methods.any?
|
||||||
|
raise MethodOverrideError, "Filter overrides registered public methods as non public: #{invokable_non_public_methods.join(', ')}"
|
||||||
|
end
|
||||||
|
|
||||||
|
include(filter)
|
||||||
|
|
||||||
|
filter_methods.merge(filter.public_instance_methods.map(&:to_s))
|
||||||
|
end
|
||||||
|
|
||||||
|
def invokable?(method)
|
||||||
|
filter_methods.include?(method.to_s)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def filter_methods
|
||||||
|
@filter_methods ||= Set.new
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def invoke(method, *args)
|
||||||
|
if self.class.invokable?(method)
|
||||||
|
send(method, *args)
|
||||||
|
elsif @context.strict_filters
|
||||||
|
raise Liquid::UndefinedFilter, "undefined filter #{method}"
|
||||||
|
else
|
||||||
|
args.first
|
||||||
|
end
|
||||||
|
rescue ::ArgumentError => e
|
||||||
|
raise Liquid::ArgumentError, e.message, e.backtrace
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -13,15 +13,13 @@ module Liquid
|
|||||||
tag
|
tag
|
||||||
end
|
end
|
||||||
|
|
||||||
def disable_tags(*tags)
|
def disable_tags(*tag_names)
|
||||||
disabled_tags.push(*tags)
|
@disabled_tags ||= []
|
||||||
|
@disabled_tags.concat(tag_names)
|
||||||
|
prepend(Disabler)
|
||||||
end
|
end
|
||||||
|
|
||||||
private :new
|
private :new
|
||||||
|
|
||||||
def disabled_tags
|
|
||||||
@disabled_tags ||= []
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def initialize(tag_name, markup, parse_context)
|
def initialize(tag_name, markup, parse_context)
|
||||||
@@ -46,14 +44,6 @@ module Liquid
|
|||||||
''
|
''
|
||||||
end
|
end
|
||||||
|
|
||||||
def disabled?(context)
|
|
||||||
context.registers[:disabled_tags].disabled?(tag_name)
|
|
||||||
end
|
|
||||||
|
|
||||||
def disabled_error_message
|
|
||||||
"#{tag_name} #{options[:locale].t('errors.disabled.tag')}"
|
|
||||||
end
|
|
||||||
|
|
||||||
# For backwards compatibility with custom tags. In a future release, the semantics
|
# For backwards compatibility with custom tags. In a future release, the semantics
|
||||||
# of the `render_to_output_buffer` method will become the default and the `render`
|
# of the `render_to_output_buffer` method will become the default and the `render`
|
||||||
# method will be removed.
|
# method will be removed.
|
||||||
@@ -66,8 +56,10 @@ module Liquid
|
|||||||
false
|
false
|
||||||
end
|
end
|
||||||
|
|
||||||
def disabled_tags
|
private
|
||||||
self.class.disabled_tags
|
|
||||||
|
def parse_expression(markup)
|
||||||
|
parse_context.parse_expression(markup)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
22
lib/liquid/tag/disableable.rb
Normal file
22
lib/liquid/tag/disableable.rb
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Liquid
|
||||||
|
class Tag
|
||||||
|
module Disableable
|
||||||
|
def render_to_output_buffer(context, output)
|
||||||
|
if context.tag_disabled?(tag_name)
|
||||||
|
output << disabled_error(context)
|
||||||
|
return
|
||||||
|
end
|
||||||
|
super
|
||||||
|
end
|
||||||
|
|
||||||
|
def disabled_error(context)
|
||||||
|
# raise then rescue the exception so that the Context#exception_renderer can re-raise it
|
||||||
|
raise DisabledError, "#{tag_name} #{parse_context[:locale].t('errors.disabled.tag')}"
|
||||||
|
rescue DisabledError => exc
|
||||||
|
context.handle_error(exc, line_number)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
21
lib/liquid/tag/disabler.rb
Normal file
21
lib/liquid/tag/disabler.rb
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Liquid
|
||||||
|
class Tag
|
||||||
|
module Disabler
|
||||||
|
module ClassMethods
|
||||||
|
attr_reader :disabled_tags
|
||||||
|
end
|
||||||
|
|
||||||
|
def self.prepended(base)
|
||||||
|
base.extend(ClassMethods)
|
||||||
|
end
|
||||||
|
|
||||||
|
def render_to_output_buffer(context, output)
|
||||||
|
context.with_disabled_tags(self.class.disabled_tags) do
|
||||||
|
super
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -12,26 +12,27 @@ module Liquid
|
|||||||
class Assign < Tag
|
class Assign < Tag
|
||||||
Syntax = /(#{VariableSignature}+)\s*=\s*(.*)\s*/om
|
Syntax = /(#{VariableSignature}+)\s*=\s*(.*)\s*/om
|
||||||
|
|
||||||
def self.syntax_error_translation_key
|
# @api private
|
||||||
"errors.syntax.assign"
|
def self.raise_syntax_error(parse_context)
|
||||||
|
raise Liquid::SyntaxError, parse_context.locale.t('errors.syntax.assign')
|
||||||
end
|
end
|
||||||
|
|
||||||
attr_reader :to, :from
|
attr_reader :to, :from
|
||||||
|
|
||||||
def initialize(tag_name, markup, options)
|
def initialize(tag_name, markup, parse_context)
|
||||||
super
|
super
|
||||||
if markup =~ Syntax
|
if markup =~ Syntax
|
||||||
@to = Regexp.last_match(1)
|
@to = Regexp.last_match(1)
|
||||||
@from = Variable.new(Regexp.last_match(2), options)
|
@from = Variable.new(Regexp.last_match(2), parse_context)
|
||||||
else
|
else
|
||||||
raise SyntaxError, options[:locale].t(self.class.syntax_error_translation_key)
|
self.class.raise_syntax_error(parse_context)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def render_to_output_buffer(context, output)
|
def render_to_output_buffer(context, output)
|
||||||
val = @from.render(context)
|
val = @from.render(context)
|
||||||
context.scopes.last[@to] = val
|
context.scopes.last[@to] = val
|
||||||
context.resource_limits.assign_score += assign_score_of(val)
|
context.resource_limits.increment_assign_score(assign_score_of(val))
|
||||||
output
|
output
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -44,11 +45,18 @@ module Liquid
|
|||||||
def assign_score_of(val)
|
def assign_score_of(val)
|
||||||
if val.instance_of?(String)
|
if val.instance_of?(String)
|
||||||
val.bytesize
|
val.bytesize
|
||||||
elsif val.instance_of?(Array) || val.instance_of?(Hash)
|
elsif val.instance_of?(Array)
|
||||||
sum = 1
|
sum = 1
|
||||||
# Uses #each to avoid extra allocations.
|
# Uses #each to avoid extra allocations.
|
||||||
val.each { |child| sum += assign_score_of(child) }
|
val.each { |child| sum += assign_score_of(child) }
|
||||||
sum
|
sum
|
||||||
|
elsif val.instance_of?(Hash)
|
||||||
|
sum = 1
|
||||||
|
val.each do |key, entry_value|
|
||||||
|
sum += assign_score_of(key)
|
||||||
|
sum += assign_score_of(entry_value)
|
||||||
|
end
|
||||||
|
sum
|
||||||
else
|
else
|
||||||
1
|
1
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -11,8 +11,11 @@ module Liquid
|
|||||||
# {% endfor %}
|
# {% endfor %}
|
||||||
#
|
#
|
||||||
class Break < Tag
|
class Break < Tag
|
||||||
def interrupt
|
INTERRUPT = BreakInterrupt.new.freeze
|
||||||
BreakInterrupt.new
|
|
||||||
|
def render_to_output_buffer(context, output)
|
||||||
|
context.push_interrupt(INTERRUPT)
|
||||||
|
output
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -25,10 +25,10 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def render_to_output_buffer(context, output)
|
def render_to_output_buffer(context, output)
|
||||||
previous_output_size = output.bytesize
|
context.resource_limits.with_capture do
|
||||||
super
|
capture_output = render(context)
|
||||||
context.scopes.last[@to] = output
|
context.scopes.last[@to] = capture_output
|
||||||
context.resource_limits.assign_score += (output.bytesize - previous_output_size)
|
end
|
||||||
output
|
output
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -12,15 +12,23 @@ module Liquid
|
|||||||
@blocks = []
|
@blocks = []
|
||||||
|
|
||||||
if markup =~ Syntax
|
if markup =~ Syntax
|
||||||
@left = Expression.parse(Regexp.last_match(1))
|
@left = parse_expression(Regexp.last_match(1))
|
||||||
else
|
else
|
||||||
raise SyntaxError, options[:locale].t("errors.syntax.case")
|
raise SyntaxError, options[:locale].t("errors.syntax.case")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse(tokens)
|
def parse(tokens)
|
||||||
body = BlockBody.new
|
body = case_body = new_body
|
||||||
body = @blocks.last.attachment while parse_body(body, tokens)
|
body = @blocks.last.attachment while parse_body(body, tokens)
|
||||||
|
@blocks.reverse_each do |condition|
|
||||||
|
body = condition.attachment
|
||||||
|
unless body.frozen?
|
||||||
|
body.remove_blank_strings if blank?
|
||||||
|
body.freeze
|
||||||
|
end
|
||||||
|
end
|
||||||
|
case_body.freeze
|
||||||
end
|
end
|
||||||
|
|
||||||
def nodelist
|
def nodelist
|
||||||
@@ -56,7 +64,7 @@ module Liquid
|
|||||||
private
|
private
|
||||||
|
|
||||||
def record_when_condition(markup)
|
def record_when_condition(markup)
|
||||||
body = BlockBody.new
|
body = new_body
|
||||||
|
|
||||||
while markup
|
while markup
|
||||||
unless markup =~ WhenSyntax
|
unless markup =~ WhenSyntax
|
||||||
@@ -65,7 +73,7 @@ module Liquid
|
|||||||
|
|
||||||
markup = Regexp.last_match(2)
|
markup = Regexp.last_match(2)
|
||||||
|
|
||||||
block = Condition.new(@left, '==', Expression.parse(Regexp.last_match(1)))
|
block = Condition.new(@left, '==', Condition.parse_expression(parse_context, Regexp.last_match(1)))
|
||||||
block.attach(body)
|
block.attach(body)
|
||||||
@blocks << block
|
@blocks << block
|
||||||
end
|
end
|
||||||
@@ -77,7 +85,7 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
block = ElseCondition.new
|
block = ElseCondition.new
|
||||||
block.attach(BlockBody.new)
|
block.attach(new_body)
|
||||||
@blocks << block
|
@blocks << block
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -11,8 +11,11 @@ module Liquid
|
|||||||
# {% endfor %}
|
# {% endfor %}
|
||||||
#
|
#
|
||||||
class Continue < Tag
|
class Continue < Tag
|
||||||
def interrupt
|
INTERRUPT = ContinueInterrupt.new.freeze
|
||||||
ContinueInterrupt.new
|
|
||||||
|
def render_to_output_buffer(context, output)
|
||||||
|
context.push_interrupt(INTERRUPT)
|
||||||
|
output
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ module Liquid
|
|||||||
case markup
|
case markup
|
||||||
when NamedSyntax
|
when NamedSyntax
|
||||||
@variables = variables_from_string(Regexp.last_match(2))
|
@variables = variables_from_string(Regexp.last_match(2))
|
||||||
@name = Expression.parse(Regexp.last_match(1))
|
@name = parse_expression(Regexp.last_match(1))
|
||||||
when SimpleSyntax
|
when SimpleSyntax
|
||||||
@variables = variables_from_string(markup)
|
@variables = variables_from_string(markup)
|
||||||
@name = @variables.to_s
|
@name = @variables.to_s
|
||||||
@@ -51,8 +51,8 @@ module Liquid
|
|||||||
|
|
||||||
iteration += 1
|
iteration += 1
|
||||||
iteration = 0 if iteration >= @variables.size
|
iteration = 0 if iteration >= @variables.size
|
||||||
context.registers[:cycle][key] = iteration
|
|
||||||
|
|
||||||
|
context.registers[:cycle][key] = iteration
|
||||||
output
|
output
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -61,7 +61,7 @@ module Liquid
|
|||||||
def variables_from_string(markup)
|
def variables_from_string(markup)
|
||||||
markup.split(',').collect do |var|
|
markup.split(',').collect do |var|
|
||||||
var =~ /\s*(#{QuotedFragment})\s*/o
|
var =~ /\s*(#{QuotedFragment})\s*/o
|
||||||
Regexp.last_match(1) ? Expression.parse(Regexp.last_match(1)) : nil
|
Regexp.last_match(1) ? parse_expression(Regexp.last_match(1)) : nil
|
||||||
end.compact
|
end.compact
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -54,14 +54,21 @@ module Liquid
|
|||||||
super
|
super
|
||||||
@from = @limit = nil
|
@from = @limit = nil
|
||||||
parse_with_selected_parser(markup)
|
parse_with_selected_parser(markup)
|
||||||
@for_block = BlockBody.new
|
@for_block = new_body
|
||||||
@else_block = nil
|
@else_block = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse(tokens)
|
def parse(tokens)
|
||||||
return unless parse_body(@for_block, tokens)
|
if parse_body(@for_block, tokens)
|
||||||
parse_body(@else_block, tokens)
|
parse_body(@else_block, tokens)
|
||||||
end
|
end
|
||||||
|
if blank?
|
||||||
|
@else_block&.remove_blank_strings
|
||||||
|
@for_block.remove_blank_strings
|
||||||
|
end
|
||||||
|
@else_block&.freeze
|
||||||
|
@for_block.freeze
|
||||||
|
end
|
||||||
|
|
||||||
def nodelist
|
def nodelist
|
||||||
@else_block ? [@for_block, @else_block] : [@for_block]
|
@else_block ? [@for_block, @else_block] : [@for_block]
|
||||||
@@ -69,7 +76,7 @@ module Liquid
|
|||||||
|
|
||||||
def unknown_tag(tag, markup, tokens)
|
def unknown_tag(tag, markup, tokens)
|
||||||
return super unless tag == 'else'
|
return super unless tag == 'else'
|
||||||
@else_block = BlockBody.new
|
@else_block = new_body
|
||||||
end
|
end
|
||||||
|
|
||||||
def render_to_output_buffer(context, output)
|
def render_to_output_buffer(context, output)
|
||||||
@@ -92,7 +99,7 @@ module Liquid
|
|||||||
collection_name = Regexp.last_match(2)
|
collection_name = Regexp.last_match(2)
|
||||||
@reversed = !!Regexp.last_match(3)
|
@reversed = !!Regexp.last_match(3)
|
||||||
@name = "#{@variable_name}-#{collection_name}"
|
@name = "#{@variable_name}-#{collection_name}"
|
||||||
@collection_name = Expression.parse(collection_name)
|
@collection_name = parse_expression(collection_name)
|
||||||
markup.scan(TagAttributes) do |key, value|
|
markup.scan(TagAttributes) do |key, value|
|
||||||
set_attribute(key, value)
|
set_attribute(key, value)
|
||||||
end
|
end
|
||||||
@@ -105,9 +112,11 @@ module Liquid
|
|||||||
p = Parser.new(markup)
|
p = Parser.new(markup)
|
||||||
@variable_name = p.consume(:id)
|
@variable_name = p.consume(:id)
|
||||||
raise SyntaxError, options[:locale].t("errors.syntax.for_invalid_in") unless p.id?('in')
|
raise SyntaxError, options[:locale].t("errors.syntax.for_invalid_in") unless p.id?('in')
|
||||||
|
|
||||||
collection_name = p.expression
|
collection_name = p.expression
|
||||||
|
@collection_name = parse_expression(collection_name)
|
||||||
|
|
||||||
@name = "#{@variable_name}-#{collection_name}"
|
@name = "#{@variable_name}-#{collection_name}"
|
||||||
@collection_name = Expression.parse(collection_name)
|
|
||||||
@reversed = p.id?('reversed')
|
@reversed = p.id?('reversed')
|
||||||
|
|
||||||
while p.look(:id) && p.look(:colon, 1)
|
while p.look(:id) && p.look(:colon, 1)
|
||||||
@@ -189,12 +198,13 @@ module Liquid
|
|||||||
case key
|
case key
|
||||||
when 'offset'
|
when 'offset'
|
||||||
@from = if expr == 'continue'
|
@from = if expr == 'continue'
|
||||||
|
Usage.increment('for_offset_continue')
|
||||||
:continue
|
:continue
|
||||||
else
|
else
|
||||||
Expression.parse(expr)
|
parse_expression(expr)
|
||||||
end
|
end
|
||||||
when 'limit'
|
when 'limit'
|
||||||
@limit = Expression.parse(expr)
|
@limit = parse_expression(expr)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -31,10 +31,17 @@ module Liquid
|
|||||||
def parse(tokens)
|
def parse(tokens)
|
||||||
while parse_body(@blocks.last.attachment, tokens)
|
while parse_body(@blocks.last.attachment, tokens)
|
||||||
end
|
end
|
||||||
|
@blocks.reverse_each do |block|
|
||||||
|
block.attachment.remove_blank_strings if blank?
|
||||||
|
block.attachment.freeze
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
ELSE_TAG_NAMES = ['elsif', 'else'].freeze
|
||||||
|
private_constant :ELSE_TAG_NAMES
|
||||||
|
|
||||||
def unknown_tag(tag, markup, tokens)
|
def unknown_tag(tag, markup, tokens)
|
||||||
if ['elsif', 'else'].include?(tag)
|
if ELSE_TAG_NAMES.include?(tag)
|
||||||
push_block(tag, markup)
|
push_block(tag, markup)
|
||||||
else
|
else
|
||||||
super
|
super
|
||||||
@@ -61,21 +68,25 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
@blocks.push(block)
|
@blocks.push(block)
|
||||||
block.attach(BlockBody.new)
|
block.attach(new_body)
|
||||||
|
end
|
||||||
|
|
||||||
|
def parse_expression(markup)
|
||||||
|
Condition.parse_expression(parse_context, markup)
|
||||||
end
|
end
|
||||||
|
|
||||||
def lax_parse(markup)
|
def lax_parse(markup)
|
||||||
expressions = markup.scan(ExpressionsAndOperators)
|
expressions = markup.scan(ExpressionsAndOperators)
|
||||||
raise SyntaxError, options[:locale].t("errors.syntax.if") unless expressions.pop =~ Syntax
|
raise SyntaxError, options[:locale].t("errors.syntax.if") unless expressions.pop =~ Syntax
|
||||||
|
|
||||||
condition = Condition.new(Expression.parse(Regexp.last_match(1)), Regexp.last_match(2), Expression.parse(Regexp.last_match(3)))
|
condition = Condition.new(parse_expression(Regexp.last_match(1)), Regexp.last_match(2), parse_expression(Regexp.last_match(3)))
|
||||||
|
|
||||||
until expressions.empty?
|
until expressions.empty?
|
||||||
operator = expressions.pop.to_s.strip
|
operator = expressions.pop.to_s.strip
|
||||||
|
|
||||||
raise SyntaxError, options[:locale].t("errors.syntax.if") unless expressions.pop.to_s =~ Syntax
|
raise SyntaxError, options[:locale].t("errors.syntax.if") unless expressions.pop.to_s =~ Syntax
|
||||||
|
|
||||||
new_condition = Condition.new(Expression.parse(Regexp.last_match(1)), Regexp.last_match(2), Expression.parse(Regexp.last_match(3)))
|
new_condition = Condition.new(parse_expression(Regexp.last_match(1)), Regexp.last_match(2), parse_expression(Regexp.last_match(3)))
|
||||||
raise SyntaxError, options[:locale].t("errors.syntax.if") unless BOOLEAN_OPERATORS.include?(operator)
|
raise SyntaxError, options[:locale].t("errors.syntax.if") unless BOOLEAN_OPERATORS.include?(operator)
|
||||||
new_condition.send(operator, condition)
|
new_condition.send(operator, condition)
|
||||||
condition = new_condition
|
condition = new_condition
|
||||||
@@ -103,9 +114,9 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def parse_comparison(p)
|
def parse_comparison(p)
|
||||||
a = Expression.parse(p.expression)
|
a = parse_expression(p.expression)
|
||||||
if (op = p.consume?(:comparison))
|
if (op = p.consume?(:comparison))
|
||||||
b = Expression.parse(p.expression)
|
b = parse_expression(p.expression)
|
||||||
Condition.new(a, op, b)
|
Condition.new(a, op, b)
|
||||||
else
|
else
|
||||||
Condition.new(a)
|
Condition.new(a)
|
||||||
|
|||||||
@@ -16,24 +16,28 @@ module Liquid
|
|||||||
# {% include 'product' for products %}
|
# {% include 'product' for products %}
|
||||||
#
|
#
|
||||||
class Include < Tag
|
class Include < Tag
|
||||||
Syntax = /(#{QuotedFragment}+)(\s+(?:with|for)\s+(#{QuotedFragment}+))?/o
|
prepend Tag::Disableable
|
||||||
|
|
||||||
|
SYNTAX = /(#{QuotedFragment}+)(\s+(?:with|for)\s+(#{QuotedFragment}+))?(\s+(?:as)\s+(#{VariableSegment}+))?/o
|
||||||
|
Syntax = SYNTAX
|
||||||
|
|
||||||
attr_reader :template_name_expr, :variable_name_expr, :attributes
|
attr_reader :template_name_expr, :variable_name_expr, :attributes
|
||||||
|
|
||||||
def initialize(tag_name, markup, options)
|
def initialize(tag_name, markup, options)
|
||||||
super
|
super
|
||||||
|
|
||||||
if markup =~ Syntax
|
if markup =~ SYNTAX
|
||||||
|
|
||||||
template_name = Regexp.last_match(1)
|
template_name = Regexp.last_match(1)
|
||||||
variable_name = Regexp.last_match(3)
|
variable_name = Regexp.last_match(3)
|
||||||
|
|
||||||
@variable_name_expr = variable_name ? Expression.parse(variable_name) : nil
|
@alias_name = Regexp.last_match(5)
|
||||||
@template_name_expr = Expression.parse(template_name)
|
@variable_name_expr = variable_name ? parse_expression(variable_name) : nil
|
||||||
|
@template_name_expr = parse_expression(template_name)
|
||||||
@attributes = {}
|
@attributes = {}
|
||||||
|
|
||||||
markup.scan(TagAttributes) do |key, value|
|
markup.scan(TagAttributes) do |key, value|
|
||||||
@attributes[key] = Expression.parse(value)
|
@attributes[key] = parse_expression(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
else
|
else
|
||||||
@@ -54,7 +58,7 @@ module Liquid
|
|||||||
parse_context: parse_context
|
parse_context: parse_context
|
||||||
)
|
)
|
||||||
|
|
||||||
context_variable_name = template_name.split('/').last
|
context_variable_name = @alias_name || template_name.split('/').last
|
||||||
|
|
||||||
variable = if @variable_name_expr
|
variable = if @variable_name_expr
|
||||||
context.evaluate(@variable_name_expr)
|
context.evaluate(@variable_name_expr)
|
||||||
|
|||||||
25
lib/liquid/tags/inline_comment.rb
Normal file
25
lib/liquid/tags/inline_comment.rb
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Liquid
|
||||||
|
class InlineComment < Tag
|
||||||
|
def initialize(tag_name, markup, options)
|
||||||
|
super
|
||||||
|
# Semantically, a comment should only ignore everything after it on the line.
|
||||||
|
# Currently, this implementation doesn't support mixing a comment with another tag
|
||||||
|
# but we need to reserve future support for this.
|
||||||
|
if markup.match?(/\n\s*[^#]/)
|
||||||
|
raise SyntaxError, options[:locale].t("errors.syntax.inline_comment_invalid")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def render_to_output_buffer(_context, output)
|
||||||
|
output
|
||||||
|
end
|
||||||
|
|
||||||
|
def blank?
|
||||||
|
true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
Template.register_tag('#', InlineComment)
|
||||||
|
end
|
||||||
@@ -14,14 +14,14 @@ module Liquid
|
|||||||
def parse(tokens)
|
def parse(tokens)
|
||||||
@body = +''
|
@body = +''
|
||||||
while (token = tokens.shift)
|
while (token = tokens.shift)
|
||||||
if token =~ FullTokenPossiblyInvalid
|
if token =~ FullTokenPossiblyInvalid && block_delimiter == Regexp.last_match(2)
|
||||||
@body << Regexp.last_match(1) if Regexp.last_match(1) != ""
|
@body << Regexp.last_match(1) if Regexp.last_match(1) != ""
|
||||||
return if block_delimiter == Regexp.last_match(2)
|
return
|
||||||
end
|
end
|
||||||
@body << token unless token.empty?
|
@body << token unless token.empty?
|
||||||
end
|
end
|
||||||
|
|
||||||
raise SyntaxError, parse_context.locale.t("errors.syntax.tag_never_closed", block_name: block_name)
|
raise_tag_never_closed(block_name)
|
||||||
end
|
end
|
||||||
|
|
||||||
def render_to_output_buffer(_context, output)
|
def render_to_output_buffer(_context, output)
|
||||||
|
|||||||
@@ -2,7 +2,8 @@
|
|||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
class Render < Tag
|
class Render < Tag
|
||||||
SYNTAX = /(#{QuotedString})#{QuotedFragment}*/o
|
FOR = 'for'
|
||||||
|
SYNTAX = /(#{QuotedString}+)(\s+(with|#{FOR})\s+(#{QuotedFragment}+))?(\s+(?:as)\s+(#{VariableSegment}+))?/o
|
||||||
|
|
||||||
disable_tags "include"
|
disable_tags "include"
|
||||||
|
|
||||||
@@ -14,12 +15,17 @@ module Liquid
|
|||||||
raise SyntaxError, options[:locale].t("errors.syntax.render") unless markup =~ SYNTAX
|
raise SyntaxError, options[:locale].t("errors.syntax.render") unless markup =~ SYNTAX
|
||||||
|
|
||||||
template_name = Regexp.last_match(1)
|
template_name = Regexp.last_match(1)
|
||||||
|
with_or_for = Regexp.last_match(3)
|
||||||
|
variable_name = Regexp.last_match(4)
|
||||||
|
|
||||||
@template_name_expr = Expression.parse(template_name)
|
@alias_name = Regexp.last_match(6)
|
||||||
|
@variable_name_expr = variable_name ? parse_expression(variable_name) : nil
|
||||||
|
@template_name_expr = parse_expression(template_name)
|
||||||
|
@for = (with_or_for == FOR)
|
||||||
|
|
||||||
@attributes = {}
|
@attributes = {}
|
||||||
markup.scan(TagAttributes) do |key, value|
|
markup.scan(TagAttributes) do |key, value|
|
||||||
@attributes[key] = Expression.parse(value)
|
@attributes[key] = parse_expression(value)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -38,13 +44,29 @@ module Liquid
|
|||||||
parse_context: parse_context
|
parse_context: parse_context
|
||||||
)
|
)
|
||||||
|
|
||||||
|
context_variable_name = @alias_name || template_name.split('/').last
|
||||||
|
|
||||||
|
render_partial_func = ->(var, forloop) {
|
||||||
inner_context = context.new_isolated_subcontext
|
inner_context = context.new_isolated_subcontext
|
||||||
inner_context.template_name = template_name
|
inner_context.template_name = template_name
|
||||||
inner_context.partial = true
|
inner_context.partial = true
|
||||||
|
inner_context['forloop'] = forloop if forloop
|
||||||
|
|
||||||
@attributes.each do |key, value|
|
@attributes.each do |key, value|
|
||||||
inner_context[key] = context.evaluate(value)
|
inner_context[key] = context.evaluate(value)
|
||||||
end
|
end
|
||||||
|
inner_context[context_variable_name] = var unless var.nil?
|
||||||
partial.render_to_output_buffer(inner_context, output)
|
partial.render_to_output_buffer(inner_context, output)
|
||||||
|
forloop&.send(:increment!)
|
||||||
|
}
|
||||||
|
|
||||||
|
variable = @variable_name_expr ? context.evaluate(@variable_name_expr) : nil
|
||||||
|
if @for && variable.respond_to?(:each) && variable.respond_to?(:count)
|
||||||
|
forloop = Liquid::ForloopDrop.new(template_name, variable.count, nil)
|
||||||
|
variable.each { |var| render_partial_func.call(var, forloop) }
|
||||||
|
else
|
||||||
|
render_partial_func.call(variable, nil)
|
||||||
|
end
|
||||||
|
|
||||||
output
|
output
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -10,10 +10,10 @@ module Liquid
|
|||||||
super
|
super
|
||||||
if markup =~ Syntax
|
if markup =~ Syntax
|
||||||
@variable_name = Regexp.last_match(1)
|
@variable_name = Regexp.last_match(1)
|
||||||
@collection_name = Expression.parse(Regexp.last_match(2))
|
@collection_name = parse_expression(Regexp.last_match(2))
|
||||||
@attributes = {}
|
@attributes = {}
|
||||||
markup.scan(TagAttributes) do |key, value|
|
markup.scan(TagAttributes) do |key, value|
|
||||||
@attributes[key] = Expression.parse(value)
|
@attributes[key] = parse_expression(value)
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
raise SyntaxError, options[:locale].t("errors.syntax.table_row")
|
raise SyntaxError, options[:locale].t("errors.syntax.table_row")
|
||||||
@@ -27,7 +27,6 @@ module Liquid
|
|||||||
to = @attributes.key?('limit') ? from + context.evaluate(@attributes['limit']).to_i : nil
|
to = @attributes.key?('limit') ? from + context.evaluate(@attributes['limit']).to_i : nil
|
||||||
|
|
||||||
collection = Utils.slice_collection(collection, from, to)
|
collection = Utils.slice_collection(collection, from, to)
|
||||||
|
|
||||||
length = collection.length
|
length = collection.length
|
||||||
|
|
||||||
cols = context.evaluate(@attributes['cols']).to_i
|
cols = context.evaluate(@attributes['cols']).to_i
|
||||||
|
|||||||
@@ -18,8 +18,6 @@ module Liquid
|
|||||||
attr_accessor :root
|
attr_accessor :root
|
||||||
attr_reader :resource_limits, :warnings
|
attr_reader :resource_limits, :warnings
|
||||||
|
|
||||||
@@file_system = BlankFileSystem.new
|
|
||||||
|
|
||||||
class TagRegistry
|
class TagRegistry
|
||||||
include Enumerable
|
include Enumerable
|
||||||
|
|
||||||
@@ -63,84 +61,54 @@ module Liquid
|
|||||||
# :lax acts like liquid 2.5 and silently ignores malformed tags in most cases.
|
# :lax acts like liquid 2.5 and silently ignores malformed tags in most cases.
|
||||||
# :warn is the default and will give deprecation warnings when invalid syntax is used.
|
# :warn is the default and will give deprecation warnings when invalid syntax is used.
|
||||||
# :strict will enforce correct syntax.
|
# :strict will enforce correct syntax.
|
||||||
attr_writer :error_mode
|
attr_accessor :error_mode
|
||||||
|
Template.error_mode = :lax
|
||||||
# Sets how strict the taint checker should be.
|
|
||||||
# :lax is the default, and ignores the taint flag completely
|
|
||||||
# :warn adds a warning, but does not interrupt the rendering
|
|
||||||
# :error raises an error when tainted output is used
|
|
||||||
attr_writer :taint_mode
|
|
||||||
|
|
||||||
attr_accessor :default_exception_renderer
|
attr_accessor :default_exception_renderer
|
||||||
Template.default_exception_renderer = lambda do |exception|
|
Template.default_exception_renderer = lambda do |exception|
|
||||||
exception
|
exception
|
||||||
end
|
end
|
||||||
|
|
||||||
def file_system
|
attr_accessor :file_system
|
||||||
@@file_system
|
Template.file_system = BlankFileSystem.new
|
||||||
end
|
|
||||||
|
|
||||||
def file_system=(obj)
|
attr_accessor :tags
|
||||||
@@file_system = obj
|
Template.tags = TagRegistry.new
|
||||||
end
|
private :tags=
|
||||||
|
|
||||||
def register_tag(name, klass)
|
def register_tag(name, klass)
|
||||||
tags[name.to_s] = klass
|
tags[name.to_s] = klass
|
||||||
end
|
end
|
||||||
|
|
||||||
def tags
|
|
||||||
@tags ||= TagRegistry.new
|
|
||||||
end
|
|
||||||
|
|
||||||
def add_register(name, klass)
|
|
||||||
registers[name.to_sym] = klass
|
|
||||||
end
|
|
||||||
|
|
||||||
def registers
|
|
||||||
@registers ||= {}
|
|
||||||
end
|
|
||||||
|
|
||||||
def error_mode
|
|
||||||
@error_mode ||= :lax
|
|
||||||
end
|
|
||||||
|
|
||||||
def taint_mode
|
|
||||||
@taint_mode ||= :lax
|
|
||||||
end
|
|
||||||
|
|
||||||
# Pass a module with filter methods which should be available
|
# Pass a module with filter methods which should be available
|
||||||
# to all liquid views. Good for registering the standard library
|
# to all liquid views. Good for registering the standard library
|
||||||
def register_filter(mod)
|
def register_filter(mod)
|
||||||
Strainer.global_filter(mod)
|
StrainerFactory.add_global_filter(mod)
|
||||||
end
|
end
|
||||||
|
|
||||||
def default_resource_limits
|
attr_accessor :default_resource_limits
|
||||||
@default_resource_limits ||= {}
|
Template.default_resource_limits = {}
|
||||||
end
|
private :default_resource_limits=
|
||||||
|
|
||||||
# creates a new <tt>Template</tt> object from liquid source code
|
# creates a new <tt>Template</tt> object from liquid source code
|
||||||
# To enable profiling, pass in <tt>profile: true</tt> as an option.
|
# To enable profiling, pass in <tt>profile: true</tt> as an option.
|
||||||
# See Liquid::Profiler for more information
|
# See Liquid::Profiler for more information
|
||||||
def parse(source, options = {})
|
def parse(source, options = {})
|
||||||
template = Template.new
|
new.parse(source, options)
|
||||||
template.parse(source, options)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def initialize
|
def initialize
|
||||||
@rethrow_errors = false
|
@rethrow_errors = false
|
||||||
@resource_limits = ResourceLimits.new(self.class.default_resource_limits)
|
@resource_limits = ResourceLimits.new(Template.default_resource_limits)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Parse source code.
|
# Parse source code.
|
||||||
# Returns self for easy chaining
|
# Returns self for easy chaining
|
||||||
def parse(source, options = {})
|
def parse(source, options = {})
|
||||||
@options = options
|
parse_context = configure_options(options)
|
||||||
@profiling = options[:profile]
|
tokenizer = parse_context.new_tokenizer(source, start_line_number: @line_numbers && 1)
|
||||||
@line_numbers = options[:line_numbers] || @profiling
|
@root = Document.parse(tokenizer, parse_context)
|
||||||
parse_context = options.is_a?(ParseContext) ? options : ParseContext.new(options)
|
|
||||||
@root = Document.parse(tokenize(source), parse_context)
|
|
||||||
@warnings = parse_context.warnings
|
|
||||||
self
|
self
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -182,7 +150,7 @@ module Liquid
|
|||||||
c = args.shift
|
c = args.shift
|
||||||
|
|
||||||
if @rethrow_errors
|
if @rethrow_errors
|
||||||
c.exception_renderer = ->(_e) { raise }
|
c.exception_renderer = Liquid::RAISE_EXCEPTION_LAMBDA
|
||||||
end
|
end
|
||||||
|
|
||||||
c
|
c
|
||||||
@@ -215,19 +183,16 @@ module Liquid
|
|||||||
context.add_filters(args.pop)
|
context.add_filters(args.pop)
|
||||||
end
|
end
|
||||||
|
|
||||||
Template.registers.each do |key, register|
|
|
||||||
context_register[key] = register
|
|
||||||
end
|
|
||||||
|
|
||||||
# Retrying a render resets resource usage
|
# Retrying a render resets resource usage
|
||||||
context.resource_limits.reset
|
context.resource_limits.reset
|
||||||
|
|
||||||
|
if @profiling && context.profiler.nil?
|
||||||
|
@profiler = context.profiler = Liquid::Profiler.new
|
||||||
|
end
|
||||||
|
|
||||||
begin
|
begin
|
||||||
# render the nodelist.
|
# render the nodelist.
|
||||||
# for performance reasons we get an array back here. join will make a string out of it.
|
|
||||||
with_profiling(context) do
|
|
||||||
@root.render_to_output_buffer(context, output || +'')
|
@root.render_to_output_buffer(context, output || +'')
|
||||||
end
|
|
||||||
rescue Liquid::MemoryError => e
|
rescue Liquid::MemoryError => e
|
||||||
context.handle_error(e)
|
context.handle_error(e)
|
||||||
ensure
|
ensure
|
||||||
@@ -246,25 +211,17 @@ module Liquid
|
|||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def tokenize(source)
|
def configure_options(options)
|
||||||
Tokenizer.new(source, @line_numbers)
|
if (profiling = options[:profile])
|
||||||
end
|
|
||||||
|
|
||||||
def with_profiling(context)
|
|
||||||
if @profiling && !context.partial
|
|
||||||
raise "Profiler not loaded, require 'liquid/profiler' first" unless defined?(Liquid::Profiler)
|
raise "Profiler not loaded, require 'liquid/profiler' first" unless defined?(Liquid::Profiler)
|
||||||
|
|
||||||
@profiler = Profiler.new
|
|
||||||
@profiler.start
|
|
||||||
|
|
||||||
begin
|
|
||||||
yield
|
|
||||||
ensure
|
|
||||||
@profiler.stop
|
|
||||||
end
|
|
||||||
else
|
|
||||||
yield
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@options = options
|
||||||
|
@profiling = profiling
|
||||||
|
@line_numbers = options[:line_numbers] || @profiling
|
||||||
|
parse_context = options.is_a?(ParseContext) ? options : ParseContext.new(options)
|
||||||
|
@warnings = parse_context.warnings
|
||||||
|
parse_context
|
||||||
end
|
end
|
||||||
|
|
||||||
def apply_options_to_context(context, options)
|
def apply_options_to_context(context, options)
|
||||||
|
|||||||
9
lib/liquid/template_factory.rb
Normal file
9
lib/liquid/template_factory.rb
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Liquid
|
||||||
|
class TemplateFactory
|
||||||
|
def for(_template_name)
|
||||||
|
Liquid::Template.new
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -30,7 +30,7 @@ module Liquid
|
|||||||
@parse_context = parse_context
|
@parse_context = parse_context
|
||||||
@line_number = parse_context.line_number
|
@line_number = parse_context.line_number
|
||||||
|
|
||||||
parse_with_selected_parser(markup)
|
strict_parse_with_error_mode_fallback(markup)
|
||||||
end
|
end
|
||||||
|
|
||||||
def raw
|
def raw
|
||||||
@@ -63,6 +63,8 @@ module Liquid
|
|||||||
@filters = []
|
@filters = []
|
||||||
p = Parser.new(markup)
|
p = Parser.new(markup)
|
||||||
|
|
||||||
|
return if p.look(:end_of_string)
|
||||||
|
|
||||||
@name = Expression.parse(p.expression)
|
@name = Expression.parse(p.expression)
|
||||||
while p.consume?(:pipe)
|
while p.consume?(:pipe)
|
||||||
filtername = p.consume(:id)
|
filtername = p.consume(:id)
|
||||||
@@ -81,14 +83,14 @@ module Liquid
|
|||||||
end
|
end
|
||||||
|
|
||||||
def render(context)
|
def render(context)
|
||||||
obj = @filters.inject(context.evaluate(@name)) do |output, (filter_name, filter_args, filter_kwargs)|
|
obj = context.evaluate(@name)
|
||||||
|
|
||||||
|
@filters.each do |filter_name, filter_args, filter_kwargs|
|
||||||
filter_args = evaluate_filter_expressions(context, filter_args, filter_kwargs)
|
filter_args = evaluate_filter_expressions(context, filter_args, filter_kwargs)
|
||||||
context.invoke(filter_name, output, *filter_args)
|
obj = context.invoke(filter_name, obj, *filter_args)
|
||||||
end
|
end
|
||||||
|
|
||||||
obj = context.apply_global_filter(obj)
|
context.apply_global_filter(obj)
|
||||||
taint_check(context, obj)
|
|
||||||
obj
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def render_to_output_buffer(context, output)
|
def render_to_output_buffer(context, output)
|
||||||
@@ -142,25 +144,6 @@ module Liquid
|
|||||||
parsed_args
|
parsed_args
|
||||||
end
|
end
|
||||||
|
|
||||||
def taint_check(context, obj)
|
|
||||||
return unless obj.tainted?
|
|
||||||
return if Template.taint_mode == :lax
|
|
||||||
|
|
||||||
@markup =~ QuotedFragment
|
|
||||||
name = Regexp.last_match(0)
|
|
||||||
|
|
||||||
error = TaintedError.new("variable '#{name}' is tainted and was not escaped")
|
|
||||||
error.line_number = line_number
|
|
||||||
error.template_name = context.template_name
|
|
||||||
|
|
||||||
case Template.taint_mode
|
|
||||||
when :warn
|
|
||||||
context.warnings << error
|
|
||||||
when :error
|
|
||||||
raise error
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class ParseTreeVisitor < Liquid::ParseTreeVisitor
|
class ParseTreeVisitor < Liquid::ParseTreeVisitor
|
||||||
def children
|
def children
|
||||||
[@node.name] + @node.filters.flatten
|
[@node.name] + @node.filters.flatten
|
||||||
|
|||||||
@@ -2,5 +2,5 @@
|
|||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
module Liquid
|
module Liquid
|
||||||
VERSION = "4.0.3"
|
VERSION = "5.0.0"
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -17,9 +17,11 @@ Gem::Specification.new do |s|
|
|||||||
s.license = "MIT"
|
s.license = "MIT"
|
||||||
# s.description = "A secure, non-evaling end user template engine with aesthetic markup."
|
# s.description = "A secure, non-evaling end user template engine with aesthetic markup."
|
||||||
|
|
||||||
s.required_ruby_version = ">= 2.4.0"
|
s.required_ruby_version = ">= 2.5.0"
|
||||||
s.required_rubygems_version = ">= 1.3.7"
|
s.required_rubygems_version = ">= 1.3.7"
|
||||||
|
|
||||||
|
s.metadata['allowed_push_host'] = 'https://rubygems.org'
|
||||||
|
|
||||||
s.test_files = Dir.glob("{test}/**/*")
|
s.test_files = Dir.glob("{test}/**/*")
|
||||||
s.files = Dir.glob("{lib}/**/*") + %w(LICENSE README.md)
|
s.files = Dir.glob("{lib}/**/*") + %w(LICENSE README.md)
|
||||||
|
|
||||||
@@ -27,6 +29,6 @@ Gem::Specification.new do |s|
|
|||||||
|
|
||||||
s.require_path = "lib"
|
s.require_path = "lib"
|
||||||
|
|
||||||
s.add_development_dependency('rake', '~> 11.3')
|
s.add_development_dependency('rake', '~> 13.0')
|
||||||
s.add_development_dependency('minitest')
|
s.add_development_dependency('minitest')
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -21,8 +21,8 @@ class Profiler
|
|||||||
end
|
end
|
||||||
|
|
||||||
def profile(phase, &block)
|
def profile(phase, &block)
|
||||||
print LOG_LABEL
|
print(LOG_LABEL)
|
||||||
print "#{phase}.. ".ljust(10)
|
print("#{phase}.. ".ljust(10))
|
||||||
report = MemoryProfiler.report(&block)
|
report = MemoryProfiler.report(&block)
|
||||||
puts 'Done.'
|
puts 'Done.'
|
||||||
@headings << phase.capitalize
|
@headings << phase.capitalize
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ module ShopFilter
|
|||||||
|
|
||||||
def link_to_vendor(vendor)
|
def link_to_vendor(vendor)
|
||||||
if vendor
|
if vendor
|
||||||
link_to vendor, url_for_vendor(vendor), vendor
|
link_to(vendor, url_for_vendor(vendor), vendor)
|
||||||
else
|
else
|
||||||
'Unknown Vendor'
|
'Unknown Vendor'
|
||||||
end
|
end
|
||||||
@@ -39,7 +39,7 @@ module ShopFilter
|
|||||||
|
|
||||||
def link_to_type(type)
|
def link_to_type(type)
|
||||||
if type
|
if type
|
||||||
link_to type, url_for_type(type), type
|
link_to(type, url_for_type(type), type)
|
||||||
else
|
else
|
||||||
'Unknown Vendor'
|
'Unknown Vendor'
|
||||||
end
|
end
|
||||||
@@ -60,7 +60,7 @@ module ShopFilter
|
|||||||
|
|
||||||
case style
|
case style
|
||||||
when 'original'
|
when 'original'
|
||||||
return '/files/shops/random_number/' + url
|
'/files/shops/random_number/' + url
|
||||||
when 'grande', 'large', 'medium', 'compact', 'small', 'thumb', 'icon'
|
when 'grande', 'large', 'medium', 'compact', 'small', 'thumb', 'icon'
|
||||||
"/files/shops/random_number/products/#{Regexp.last_match(1)}_#{style}.#{Regexp.last_match(2)}"
|
"/files/shops/random_number/products/#{Regexp.last_match(1)}_#{style}.#{Regexp.last_match(2)}"
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -73,10 +73,14 @@ class ThemeRunner
|
|||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
def render_layout(template, layout, assigns)
|
||||||
|
assigns['content_for_layout'] = template.render!(assigns)
|
||||||
|
layout&.render!(assigns)
|
||||||
|
end
|
||||||
|
|
||||||
def compile_and_render(template, layout, assigns, page_template, template_file)
|
def compile_and_render(template, layout, assigns, page_template, template_file)
|
||||||
compiled_test = compile_test(template, layout, assigns, page_template, template_file)
|
compiled_test = compile_test(template, layout, assigns, page_template, template_file)
|
||||||
assigns['content_for_layout'] = compiled_test[:tmpl].render!(assigns)
|
render_layout(compiled_test[:tmpl], compiled_test[:layout], compiled_test[:assigns])
|
||||||
compiled_test[:layout].render!(assigns) if layout
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def compile_all_tests
|
def compile_all_tests
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ class AssignTest < Minitest::Test
|
|||||||
END_TEMPLATE
|
END_TEMPLATE
|
||||||
template = Template.parse(template_source)
|
template = Template.parse(template_source)
|
||||||
rendered = template.render!
|
rendered = template.render!
|
||||||
assert_equal "Print this-thing", rendered.strip
|
assert_equal("Print this-thing", rendered.strip)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_assigned_variable
|
def test_assigned_variable
|
||||||
@@ -44,7 +44,74 @@ class AssignTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
with_error_mode(:lax) do
|
with_error_mode(:lax) do
|
||||||
assert Template.parse("{% assign foo = ('X' | downcase) %}")
|
assert(Template.parse("{% assign foo = ('X' | downcase) %}"))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end # AssignTest
|
|
||||||
|
def test_expression_with_whitespace_in_square_brackets
|
||||||
|
source = "{% assign r = a[ 'b' ] %}{{ r }}"
|
||||||
|
assert_template_result('result', source, 'a' => { 'b' => 'result' })
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_score_exceeding_resource_limit
|
||||||
|
t = Template.parse("{% assign foo = 42 %}{% assign bar = 23 %}")
|
||||||
|
t.resource_limits.assign_score_limit = 1
|
||||||
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
|
assert(t.resource_limits.reached?)
|
||||||
|
|
||||||
|
t.resource_limits.assign_score_limit = 2
|
||||||
|
assert_equal("", t.render!)
|
||||||
|
refute_nil(t.resource_limits.assign_score)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_score_exceeding_limit_from_composite_object
|
||||||
|
t = Template.parse("{% assign foo = 'aaaa' | reverse %}")
|
||||||
|
|
||||||
|
t.resource_limits.assign_score_limit = 3
|
||||||
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
|
assert(t.resource_limits.reached?)
|
||||||
|
|
||||||
|
t.resource_limits.assign_score_limit = 5
|
||||||
|
assert_equal("", t.render!)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_score_of_int
|
||||||
|
assert_equal(1, assign_score_of(123))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_score_of_string_counts_bytes
|
||||||
|
assert_equal(3, assign_score_of('123'))
|
||||||
|
assert_equal(5, assign_score_of('12345'))
|
||||||
|
assert_equal(9, assign_score_of('すごい'))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_score_of_array
|
||||||
|
assert_equal(1, assign_score_of([]))
|
||||||
|
assert_equal(2, assign_score_of([123]))
|
||||||
|
assert_equal(6, assign_score_of([123, 'abcd']))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_score_of_hash
|
||||||
|
assert_equal(1, assign_score_of({}))
|
||||||
|
assert_equal(5, assign_score_of('int' => 123))
|
||||||
|
assert_equal(12, assign_score_of('int' => 123, 'str' => 'abcd'))
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
class ObjectWrapperDrop < Liquid::Drop
|
||||||
|
def initialize(obj)
|
||||||
|
@obj = obj
|
||||||
|
end
|
||||||
|
|
||||||
|
def value
|
||||||
|
@obj
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def assign_score_of(obj)
|
||||||
|
context = Liquid::Context.new('drop' => ObjectWrapperDrop.new(obj))
|
||||||
|
Liquid::Template.parse('{% assign obj = drop.value %}').render!(context)
|
||||||
|
context.resource_limits.assign_score
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|||||||
@@ -96,9 +96,9 @@ class BlankTest < Minitest::Test
|
|||||||
|
|
||||||
def test_include_is_blank
|
def test_include_is_blank
|
||||||
Liquid::Template.file_system = BlankTestFileSystem.new
|
Liquid::Template.file_system = BlankTestFileSystem.new
|
||||||
assert_template_result "foobar" * (N + 1), wrap("{% include 'foobar' %}")
|
assert_template_result("foobar" * (N + 1), wrap("{% include 'foobar' %}"))
|
||||||
assert_template_result " foobar " * (N + 1), wrap("{% include ' foobar ' %}")
|
assert_template_result(" foobar " * (N + 1), wrap("{% include ' foobar ' %}"))
|
||||||
assert_template_result " " * (N + 1), wrap(" {% include ' ' %} ")
|
assert_template_result(" " * (N + 1), wrap(" {% include ' ' %} "))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_case_is_blank
|
def test_case_is_blank
|
||||||
|
|||||||
@@ -9,6 +9,50 @@ class BlockTest < Minitest::Test
|
|||||||
exc = assert_raises(SyntaxError) do
|
exc = assert_raises(SyntaxError) do
|
||||||
Template.parse("{% if true %}{% endunless %}")
|
Template.parse("{% if true %}{% endunless %}")
|
||||||
end
|
end
|
||||||
assert_equal exc.message, "Liquid syntax error: 'endunless' is not a valid delimiter for if tags. use endif"
|
assert_equal(exc.message, "Liquid syntax error: 'endunless' is not a valid delimiter for if tags. use endif")
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_with_custom_tag
|
||||||
|
with_custom_tag('testtag', Block) do
|
||||||
|
assert(Liquid::Template.parse("{% testtag %} {% endtesttag %}"))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_custom_block_tags_have_a_default_render_to_output_buffer_method_for_backwards_compatibility
|
||||||
|
klass1 = Class.new(Block) do
|
||||||
|
def render(*)
|
||||||
|
'hello'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
with_custom_tag('blabla', klass1) do
|
||||||
|
template = Liquid::Template.parse("{% blabla %} bla {% endblabla %}")
|
||||||
|
|
||||||
|
assert_equal('hello', template.render)
|
||||||
|
|
||||||
|
buf = +''
|
||||||
|
output = template.render({}, output: buf)
|
||||||
|
assert_equal('hello', output)
|
||||||
|
assert_equal('hello', buf)
|
||||||
|
assert_equal(buf.object_id, output.object_id)
|
||||||
|
end
|
||||||
|
|
||||||
|
klass2 = Class.new(klass1) do
|
||||||
|
def render(*)
|
||||||
|
'foo' + super + 'bar'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
with_custom_tag('blabla', klass2) do
|
||||||
|
template = Liquid::Template.parse("{% blabla %} foo {% endblabla %}")
|
||||||
|
|
||||||
|
assert_equal('foohellobar', template.render)
|
||||||
|
|
||||||
|
buf = +''
|
||||||
|
output = template.render({}, output: buf)
|
||||||
|
assert_equal('foohellobar', output)
|
||||||
|
assert_equal('foohellobar', buf)
|
||||||
|
assert_equal(buf.object_id, output.object_id)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ class CaptureTest < Minitest::Test
|
|||||||
END_TEMPLATE
|
END_TEMPLATE
|
||||||
template = Template.parse(template_source)
|
template = Template.parse(template_source)
|
||||||
rendered = template.render!
|
rendered = template.render!
|
||||||
assert_equal "Print this-thing", rendered.strip
|
assert_equal("Print this-thing", rendered.strip)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_capture_to_variable_from_outer_scope_if_existing
|
def test_capture_to_variable_from_outer_scope_if_existing
|
||||||
@@ -32,7 +32,7 @@ class CaptureTest < Minitest::Test
|
|||||||
END_TEMPLATE
|
END_TEMPLATE
|
||||||
template = Template.parse(template_source)
|
template = Template.parse(template_source)
|
||||||
rendered = template.render!
|
rendered = template.render!
|
||||||
assert_equal "test-string", rendered.gsub(/\s/, '')
|
assert_equal("test-string", rendered.gsub(/\s/, ''))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_assigning_from_capture
|
def test_assigning_from_capture
|
||||||
@@ -47,6 +47,12 @@ class CaptureTest < Minitest::Test
|
|||||||
END_TEMPLATE
|
END_TEMPLATE
|
||||||
template = Template.parse(template_source)
|
template = Template.parse(template_source)
|
||||||
rendered = template.render!
|
rendered = template.render!
|
||||||
assert_equal "3-3", rendered.gsub(/\s/, '')
|
assert_equal("3-3", rendered.gsub(/\s/, ''))
|
||||||
end
|
end
|
||||||
end # CaptureTest
|
|
||||||
|
def test_increment_assign_score_by_bytes_not_characters
|
||||||
|
t = Template.parse("{% capture foo %}すごい{% endcapture %}")
|
||||||
|
t.render!
|
||||||
|
assert_equal(9, t.resource_limits.assign_score)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|||||||
@@ -2,9 +2,597 @@
|
|||||||
|
|
||||||
require 'test_helper'
|
require 'test_helper'
|
||||||
|
|
||||||
|
class HundredCentes
|
||||||
|
def to_liquid
|
||||||
|
100
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class CentsDrop < Liquid::Drop
|
||||||
|
def amount
|
||||||
|
HundredCentes.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def non_zero?
|
||||||
|
true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class ContextSensitiveDrop < Liquid::Drop
|
||||||
|
def test
|
||||||
|
@context['test']
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class Category < Liquid::Drop
|
||||||
|
attr_accessor :name
|
||||||
|
|
||||||
|
def initialize(name)
|
||||||
|
@name = name
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_liquid
|
||||||
|
CategoryDrop.new(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class CategoryDrop
|
||||||
|
attr_accessor :category, :context
|
||||||
|
def initialize(category)
|
||||||
|
@category = category
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class CounterDrop < Liquid::Drop
|
||||||
|
def count
|
||||||
|
@count ||= 0
|
||||||
|
@count += 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class ArrayLike
|
||||||
|
def fetch(index)
|
||||||
|
end
|
||||||
|
|
||||||
|
def [](index)
|
||||||
|
@counts ||= []
|
||||||
|
@counts[index] ||= 0
|
||||||
|
@counts[index] += 1
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_liquid
|
||||||
|
self
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
class ContextTest < Minitest::Test
|
class ContextTest < Minitest::Test
|
||||||
include Liquid
|
include Liquid
|
||||||
|
|
||||||
|
def setup
|
||||||
|
@context = Liquid::Context.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_variables
|
||||||
|
@context['string'] = 'string'
|
||||||
|
assert_equal('string', @context['string'])
|
||||||
|
|
||||||
|
@context['num'] = 5
|
||||||
|
assert_equal(5, @context['num'])
|
||||||
|
|
||||||
|
@context['time'] = Time.parse('2006-06-06 12:00:00')
|
||||||
|
assert_equal(Time.parse('2006-06-06 12:00:00'), @context['time'])
|
||||||
|
|
||||||
|
@context['date'] = Date.today
|
||||||
|
assert_equal(Date.today, @context['date'])
|
||||||
|
|
||||||
|
now = Time.now
|
||||||
|
@context['datetime'] = now
|
||||||
|
assert_equal(now, @context['datetime'])
|
||||||
|
|
||||||
|
@context['bool'] = true
|
||||||
|
assert_equal(true, @context['bool'])
|
||||||
|
|
||||||
|
@context['bool'] = false
|
||||||
|
assert_equal(false, @context['bool'])
|
||||||
|
|
||||||
|
@context['nil'] = nil
|
||||||
|
assert_nil(@context['nil'])
|
||||||
|
assert_nil(@context['nil'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_variables_not_existing
|
||||||
|
assert_nil(@context['does_not_exist'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_scoping
|
||||||
|
@context.push
|
||||||
|
@context.pop
|
||||||
|
|
||||||
|
assert_raises(Liquid::ContextError) do
|
||||||
|
@context.pop
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_raises(Liquid::ContextError) do
|
||||||
|
@context.push
|
||||||
|
@context.pop
|
||||||
|
@context.pop
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_length_query
|
||||||
|
@context['numbers'] = [1, 2, 3, 4]
|
||||||
|
|
||||||
|
assert_equal(4, @context['numbers.size'])
|
||||||
|
|
||||||
|
@context['numbers'] = { 1 => 1, 2 => 2, 3 => 3, 4 => 4 }
|
||||||
|
|
||||||
|
assert_equal(4, @context['numbers.size'])
|
||||||
|
|
||||||
|
@context['numbers'] = { 1 => 1, 2 => 2, 3 => 3, 4 => 4, 'size' => 1000 }
|
||||||
|
|
||||||
|
assert_equal(1000, @context['numbers.size'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_hyphenated_variable
|
||||||
|
@context['oh-my'] = 'godz'
|
||||||
|
assert_equal('godz', @context['oh-my'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_add_filter
|
||||||
|
filter = Module.new do
|
||||||
|
def hi(output)
|
||||||
|
output + ' hi!'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context = Context.new
|
||||||
|
context.add_filters(filter)
|
||||||
|
assert_equal('hi? hi!', context.invoke(:hi, 'hi?'))
|
||||||
|
|
||||||
|
context = Context.new
|
||||||
|
assert_equal('hi?', context.invoke(:hi, 'hi?'))
|
||||||
|
|
||||||
|
context.add_filters(filter)
|
||||||
|
assert_equal('hi? hi!', context.invoke(:hi, 'hi?'))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_only_intended_filters_make_it_there
|
||||||
|
filter = Module.new do
|
||||||
|
def hi(output)
|
||||||
|
output + ' hi!'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context = Context.new
|
||||||
|
assert_equal("Wookie", context.invoke("hi", "Wookie"))
|
||||||
|
|
||||||
|
context.add_filters(filter)
|
||||||
|
assert_equal("Wookie hi!", context.invoke("hi", "Wookie"))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_add_item_in_outer_scope
|
||||||
|
@context['test'] = 'test'
|
||||||
|
@context.push
|
||||||
|
assert_equal('test', @context['test'])
|
||||||
|
@context.pop
|
||||||
|
assert_equal('test', @context['test'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_add_item_in_inner_scope
|
||||||
|
@context.push
|
||||||
|
@context['test'] = 'test'
|
||||||
|
assert_equal('test', @context['test'])
|
||||||
|
@context.pop
|
||||||
|
assert_nil(@context['test'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_hierachical_data
|
||||||
|
@context['hash'] = { "name" => 'tobi' }
|
||||||
|
assert_equal('tobi', @context['hash.name'])
|
||||||
|
assert_equal('tobi', @context['hash["name"]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_keywords
|
||||||
|
assert_equal(true, @context['true'])
|
||||||
|
assert_equal(false, @context['false'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_digits
|
||||||
|
assert_equal(100, @context['100'])
|
||||||
|
assert_equal(100.00, @context['100.00'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_strings
|
||||||
|
assert_equal("hello!", @context['"hello!"'])
|
||||||
|
assert_equal("hello!", @context["'hello!'"])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_merge
|
||||||
|
@context.merge("test" => "test")
|
||||||
|
assert_equal('test', @context['test'])
|
||||||
|
@context.merge("test" => "newvalue", "foo" => "bar")
|
||||||
|
assert_equal('newvalue', @context['test'])
|
||||||
|
assert_equal('bar', @context['foo'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_array_notation
|
||||||
|
@context['test'] = [1, 2, 3, 4, 5]
|
||||||
|
|
||||||
|
assert_equal(1, @context['test[0]'])
|
||||||
|
assert_equal(2, @context['test[1]'])
|
||||||
|
assert_equal(3, @context['test[2]'])
|
||||||
|
assert_equal(4, @context['test[3]'])
|
||||||
|
assert_equal(5, @context['test[4]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_recoursive_array_notation
|
||||||
|
@context['test'] = { 'test' => [1, 2, 3, 4, 5] }
|
||||||
|
|
||||||
|
assert_equal(1, @context['test.test[0]'])
|
||||||
|
|
||||||
|
@context['test'] = [{ 'test' => 'worked' }]
|
||||||
|
|
||||||
|
assert_equal('worked', @context['test[0].test'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_hash_to_array_transition
|
||||||
|
@context['colors'] = {
|
||||||
|
'Blue' => ['003366', '336699', '6699CC', '99CCFF'],
|
||||||
|
'Green' => ['003300', '336633', '669966', '99CC99'],
|
||||||
|
'Yellow' => ['CC9900', 'FFCC00', 'FFFF99', 'FFFFCC'],
|
||||||
|
'Red' => ['660000', '993333', 'CC6666', 'FF9999'],
|
||||||
|
}
|
||||||
|
|
||||||
|
assert_equal('003366', @context['colors.Blue[0]'])
|
||||||
|
assert_equal('FF9999', @context['colors.Red[3]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_try_first
|
||||||
|
@context['test'] = [1, 2, 3, 4, 5]
|
||||||
|
|
||||||
|
assert_equal(1, @context['test.first'])
|
||||||
|
assert_equal(5, @context['test.last'])
|
||||||
|
|
||||||
|
@context['test'] = { 'test' => [1, 2, 3, 4, 5] }
|
||||||
|
|
||||||
|
assert_equal(1, @context['test.test.first'])
|
||||||
|
assert_equal(5, @context['test.test.last'])
|
||||||
|
|
||||||
|
@context['test'] = [1]
|
||||||
|
assert_equal(1, @context['test.first'])
|
||||||
|
assert_equal(1, @context['test.last'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_access_hashes_with_hash_notation
|
||||||
|
@context['products'] = { 'count' => 5, 'tags' => ['deepsnow', 'freestyle'] }
|
||||||
|
@context['product'] = { 'variants' => [{ 'title' => 'draft151cm' }, { 'title' => 'element151cm' }] }
|
||||||
|
|
||||||
|
assert_equal(5, @context['products["count"]'])
|
||||||
|
assert_equal('deepsnow', @context['products["tags"][0]'])
|
||||||
|
assert_equal('deepsnow', @context['products["tags"].first'])
|
||||||
|
assert_equal('draft151cm', @context['product["variants"][0]["title"]'])
|
||||||
|
assert_equal('element151cm', @context['product["variants"][1]["title"]'])
|
||||||
|
assert_equal('draft151cm', @context['product["variants"][0]["title"]'])
|
||||||
|
assert_equal('element151cm', @context['product["variants"].last["title"]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_access_variable_with_hash_notation
|
||||||
|
@context['foo'] = 'baz'
|
||||||
|
@context['bar'] = 'foo'
|
||||||
|
|
||||||
|
assert_equal('baz', @context['["foo"]'])
|
||||||
|
assert_equal('baz', @context['[bar]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_access_hashes_with_hash_access_variables
|
||||||
|
@context['var'] = 'tags'
|
||||||
|
@context['nested'] = { 'var' => 'tags' }
|
||||||
|
@context['products'] = { 'count' => 5, 'tags' => ['deepsnow', 'freestyle'] }
|
||||||
|
|
||||||
|
assert_equal('deepsnow', @context['products[var].first'])
|
||||||
|
assert_equal('freestyle', @context['products[nested.var].last'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_hash_notation_only_for_hash_access
|
||||||
|
@context['array'] = [1, 2, 3, 4, 5]
|
||||||
|
@context['hash'] = { 'first' => 'Hello' }
|
||||||
|
|
||||||
|
assert_equal(1, @context['array.first'])
|
||||||
|
assert_nil(@context['array["first"]'])
|
||||||
|
assert_equal('Hello', @context['hash["first"]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_first_can_appear_in_middle_of_callchain
|
||||||
|
@context['product'] = { 'variants' => [{ 'title' => 'draft151cm' }, { 'title' => 'element151cm' }] }
|
||||||
|
|
||||||
|
assert_equal('draft151cm', @context['product.variants[0].title'])
|
||||||
|
assert_equal('element151cm', @context['product.variants[1].title'])
|
||||||
|
assert_equal('draft151cm', @context['product.variants.first.title'])
|
||||||
|
assert_equal('element151cm', @context['product.variants.last.title'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_cents
|
||||||
|
@context.merge("cents" => HundredCentes.new)
|
||||||
|
assert_equal(100, @context['cents'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_nested_cents
|
||||||
|
@context.merge("cents" => { 'amount' => HundredCentes.new })
|
||||||
|
assert_equal(100, @context['cents.amount'])
|
||||||
|
|
||||||
|
@context.merge("cents" => { 'cents' => { 'amount' => HundredCentes.new } })
|
||||||
|
assert_equal(100, @context['cents.cents.amount'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_cents_through_drop
|
||||||
|
@context.merge("cents" => CentsDrop.new)
|
||||||
|
assert_equal(100, @context['cents.amount'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_nested_cents_through_drop
|
||||||
|
@context.merge("vars" => { "cents" => CentsDrop.new })
|
||||||
|
assert_equal(100, @context['vars.cents.amount'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_drop_methods_with_question_marks
|
||||||
|
@context.merge("cents" => CentsDrop.new)
|
||||||
|
assert(@context['cents.non_zero?'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_context_from_within_drop
|
||||||
|
@context.merge("test" => '123', "vars" => ContextSensitiveDrop.new)
|
||||||
|
assert_equal('123', @context['vars.test'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_nested_context_from_within_drop
|
||||||
|
@context.merge("test" => '123', "vars" => { "local" => ContextSensitiveDrop.new })
|
||||||
|
assert_equal('123', @context['vars.local.test'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_ranges
|
||||||
|
@context.merge("test" => '5')
|
||||||
|
assert_equal((1..5), @context['(1..5)'])
|
||||||
|
assert_equal((1..5), @context['(1..test)'])
|
||||||
|
assert_equal((5..5), @context['(test..test)'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_cents_through_drop_nestedly
|
||||||
|
@context.merge("cents" => { "cents" => CentsDrop.new })
|
||||||
|
assert_equal(100, @context['cents.cents.amount'])
|
||||||
|
|
||||||
|
@context.merge("cents" => { "cents" => { "cents" => CentsDrop.new } })
|
||||||
|
assert_equal(100, @context['cents.cents.cents.amount'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_drop_with_variable_called_only_once
|
||||||
|
@context['counter'] = CounterDrop.new
|
||||||
|
|
||||||
|
assert_equal(1, @context['counter.count'])
|
||||||
|
assert_equal(2, @context['counter.count'])
|
||||||
|
assert_equal(3, @context['counter.count'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_drop_with_key_called_only_once
|
||||||
|
@context['counter'] = CounterDrop.new
|
||||||
|
|
||||||
|
assert_equal(1, @context['counter["count"]'])
|
||||||
|
assert_equal(2, @context['counter["count"]'])
|
||||||
|
assert_equal(3, @context['counter["count"]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_proc_as_variable
|
||||||
|
@context['dynamic'] = proc { 'Hello' }
|
||||||
|
|
||||||
|
assert_equal('Hello', @context['dynamic'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_lambda_as_variable
|
||||||
|
@context['dynamic'] = proc { 'Hello' }
|
||||||
|
|
||||||
|
assert_equal('Hello', @context['dynamic'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_nested_lambda_as_variable
|
||||||
|
@context['dynamic'] = { "lambda" => proc { 'Hello' } }
|
||||||
|
|
||||||
|
assert_equal('Hello', @context['dynamic.lambda'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_array_containing_lambda_as_variable
|
||||||
|
@context['dynamic'] = [1, 2, proc { 'Hello' }, 4, 5]
|
||||||
|
|
||||||
|
assert_equal('Hello', @context['dynamic[2]'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_lambda_is_called_once
|
||||||
|
@context['callcount'] = proc {
|
||||||
|
@global ||= 0
|
||||||
|
@global += 1
|
||||||
|
@global.to_s
|
||||||
|
}
|
||||||
|
|
||||||
|
assert_equal('1', @context['callcount'])
|
||||||
|
assert_equal('1', @context['callcount'])
|
||||||
|
assert_equal('1', @context['callcount'])
|
||||||
|
|
||||||
|
@global = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_nested_lambda_is_called_once
|
||||||
|
@context['callcount'] = { "lambda" => proc {
|
||||||
|
@global ||= 0
|
||||||
|
@global += 1
|
||||||
|
@global.to_s
|
||||||
|
} }
|
||||||
|
|
||||||
|
assert_equal('1', @context['callcount.lambda'])
|
||||||
|
assert_equal('1', @context['callcount.lambda'])
|
||||||
|
assert_equal('1', @context['callcount.lambda'])
|
||||||
|
|
||||||
|
@global = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_lambda_in_array_is_called_once
|
||||||
|
@context['callcount'] = [1, 2, proc {
|
||||||
|
@global ||= 0
|
||||||
|
@global += 1
|
||||||
|
@global.to_s
|
||||||
|
}, 4, 5]
|
||||||
|
|
||||||
|
assert_equal('1', @context['callcount[2]'])
|
||||||
|
assert_equal('1', @context['callcount[2]'])
|
||||||
|
assert_equal('1', @context['callcount[2]'])
|
||||||
|
|
||||||
|
@global = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_access_to_context_from_proc
|
||||||
|
@context.registers[:magic] = 345392
|
||||||
|
|
||||||
|
@context['magic'] = proc { @context.registers[:magic] }
|
||||||
|
|
||||||
|
assert_equal(345392, @context['magic'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_to_liquid_and_context_at_first_level
|
||||||
|
@context['category'] = Category.new("foobar")
|
||||||
|
assert_kind_of(CategoryDrop, @context['category'])
|
||||||
|
assert_equal(@context, @context['category'].context)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_interrupt_avoids_object_allocations
|
||||||
|
@context.interrupt? # ruby 3.0.0 allocates on the first call
|
||||||
|
assert_no_object_allocations do
|
||||||
|
@context.interrupt?
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_context_initialization_with_a_proc_in_environment
|
||||||
|
contx = Context.new([test: ->(c) { c['poutine'] }], test: :foo)
|
||||||
|
|
||||||
|
assert(contx)
|
||||||
|
assert_nil(contx['poutine'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_apply_global_filter
|
||||||
|
global_filter_proc = ->(output) { "#{output} filtered" }
|
||||||
|
|
||||||
|
context = Context.new
|
||||||
|
context.global_filter = global_filter_proc
|
||||||
|
|
||||||
|
assert_equal('hi filtered', context.apply_global_filter('hi'))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_static_environments_are_read_with_lower_priority_than_environments
|
||||||
|
context = Context.build(
|
||||||
|
static_environments: { 'shadowed' => 'static', 'unshadowed' => 'static' },
|
||||||
|
environments: { 'shadowed' => 'dynamic' }
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_equal('dynamic', context['shadowed'])
|
||||||
|
assert_equal('static', context['unshadowed'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_apply_global_filter_when_no_global_filter_exist
|
||||||
|
context = Context.new
|
||||||
|
assert_equal('hi', context.apply_global_filter('hi'))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_does_not_inherit_variables
|
||||||
|
super_context = Context.new
|
||||||
|
super_context['my_variable'] = 'some value'
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
|
||||||
|
assert_nil(subcontext['my_variable'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_inherits_static_environment
|
||||||
|
super_context = Context.build(static_environments: { 'my_environment_value' => 'my value' })
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
|
||||||
|
assert_equal('my value', subcontext['my_environment_value'])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_inherits_resource_limits
|
||||||
|
resource_limits = ResourceLimits.new({})
|
||||||
|
super_context = Context.new({}, {}, {}, false, resource_limits)
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
assert_equal(resource_limits, subcontext.resource_limits)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_inherits_exception_renderer
|
||||||
|
super_context = Context.new
|
||||||
|
super_context.exception_renderer = ->(_e) { 'my exception message' }
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
assert_equal('my exception message', subcontext.handle_error(Liquid::Error.new))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_does_not_inherit_non_static_registers
|
||||||
|
registers = {
|
||||||
|
my_register: :my_value,
|
||||||
|
}
|
||||||
|
super_context = Context.new({}, {}, StaticRegisters.new(registers))
|
||||||
|
super_context.registers[:my_register] = :my_alt_value
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
assert_equal(:my_value, subcontext.registers[:my_register])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_inherits_static_registers
|
||||||
|
super_context = Context.build(registers: { my_register: :my_value })
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
assert_equal(:my_value, subcontext.registers[:my_register])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_registers_do_not_pollute_context
|
||||||
|
super_context = Context.build(registers: { my_register: :my_value })
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
subcontext.registers[:my_register] = :my_alt_value
|
||||||
|
assert_equal(:my_value, super_context.registers[:my_register])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_new_isolated_subcontext_inherits_filters
|
||||||
|
my_filter = Module.new do
|
||||||
|
def my_filter(*)
|
||||||
|
'my filter result'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
super_context = Context.new
|
||||||
|
super_context.add_filters([my_filter])
|
||||||
|
subcontext = super_context.new_isolated_subcontext
|
||||||
|
template = Template.parse('{{ 123 | my_filter }}')
|
||||||
|
assert_equal('my filter result', template.render(subcontext))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_disables_tag_specified
|
||||||
|
context = Context.new
|
||||||
|
context.with_disabled_tags(%w(foo bar)) do
|
||||||
|
assert_equal(true, context.tag_disabled?("foo"))
|
||||||
|
assert_equal(true, context.tag_disabled?("bar"))
|
||||||
|
assert_equal(false, context.tag_disabled?("unknown"))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_disables_nested_tags
|
||||||
|
context = Context.new
|
||||||
|
context.with_disabled_tags(["foo"]) do
|
||||||
|
context.with_disabled_tags(["foo"]) do
|
||||||
|
assert_equal(true, context.tag_disabled?("foo"))
|
||||||
|
assert_equal(false, context.tag_disabled?("bar"))
|
||||||
|
end
|
||||||
|
context.with_disabled_tags(["bar"]) do
|
||||||
|
assert_equal(true, context.tag_disabled?("foo"))
|
||||||
|
assert_equal(true, context.tag_disabled?("bar"))
|
||||||
|
context.with_disabled_tags(["foo"]) do
|
||||||
|
assert_equal(true, context.tag_disabled?("foo"))
|
||||||
|
assert_equal(true, context.tag_disabled?("bar"))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
assert_equal(true, context.tag_disabled?("foo"))
|
||||||
|
assert_equal(false, context.tag_disabled?("bar"))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def test_override_global_filter
|
def test_override_global_filter
|
||||||
global = Module.new do
|
global = Module.new do
|
||||||
def notice(output)
|
def notice(output)
|
||||||
@@ -19,16 +607,30 @@ class ContextTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
with_global_filter(global) do
|
with_global_filter(global) do
|
||||||
assert_equal 'Global test', Template.parse("{{'test' | notice }}").render!
|
assert_equal('Global test', Template.parse("{{'test' | notice }}").render!)
|
||||||
assert_equal 'Local test', Template.parse("{{'test' | notice }}").render!({}, filters: [local])
|
assert_equal('Local test', Template.parse("{{'test' | notice }}").render!({}, filters: [local]))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_has_key_will_not_add_an_error_for_missing_keys
|
def test_has_key_will_not_add_an_error_for_missing_keys
|
||||||
with_error_mode :strict do
|
with_error_mode(:strict) do
|
||||||
context = Context.new
|
context = Context.new
|
||||||
context.key?('unknown')
|
context.key?('unknown')
|
||||||
assert_empty context.errors
|
assert_empty(context.errors)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def assert_no_object_allocations
|
||||||
|
unless RUBY_ENGINE == 'ruby'
|
||||||
|
skip("stackprof needed to count object allocations")
|
||||||
|
end
|
||||||
|
require 'stackprof'
|
||||||
|
|
||||||
|
profile = StackProf.run(mode: :object) do
|
||||||
|
yield
|
||||||
|
end
|
||||||
|
assert_equal(0, profile[:samples])
|
||||||
|
end
|
||||||
|
end # ContextTest
|
||||||
|
|||||||
@@ -9,13 +9,13 @@ class DocumentTest < Minitest::Test
|
|||||||
exc = assert_raises(SyntaxError) do
|
exc = assert_raises(SyntaxError) do
|
||||||
Template.parse("{% else %}")
|
Template.parse("{% else %}")
|
||||||
end
|
end
|
||||||
assert_equal exc.message, "Liquid syntax error: Unexpected outer 'else' tag"
|
assert_equal(exc.message, "Liquid syntax error: Unexpected outer 'else' tag")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_unknown_tag
|
def test_unknown_tag
|
||||||
exc = assert_raises(SyntaxError) do
|
exc = assert_raises(SyntaxError) do
|
||||||
Template.parse("{% foo %}")
|
Template.parse("{% foo %}")
|
||||||
end
|
end
|
||||||
assert_equal exc.message, "Liquid syntax error: Unknown tag 'foo'"
|
assert_equal(exc.message, "Liquid syntax error: Unknown tag 'foo'")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -49,10 +49,6 @@ class ProductDrop < Liquid::Drop
|
|||||||
ContextDrop.new
|
ContextDrop.new
|
||||||
end
|
end
|
||||||
|
|
||||||
def user_input
|
|
||||||
(+"foo").taint
|
|
||||||
end
|
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
|
||||||
def callmenot
|
def callmenot
|
||||||
@@ -111,172 +107,151 @@ class DropsTest < Minitest::Test
|
|||||||
|
|
||||||
def test_product_drop
|
def test_product_drop
|
||||||
tpl = Liquid::Template.parse(' ')
|
tpl = Liquid::Template.parse(' ')
|
||||||
assert_equal ' ', tpl.render!('product' => ProductDrop.new)
|
assert_equal(' ', tpl.render!('product' => ProductDrop.new))
|
||||||
end
|
|
||||||
|
|
||||||
def test_rendering_raises_on_tainted_attr
|
|
||||||
with_taint_mode(:error) do
|
|
||||||
tpl = Liquid::Template.parse('{{ product.user_input }}')
|
|
||||||
assert_raises TaintedError do
|
|
||||||
tpl.render!('product' => ProductDrop.new)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_rendering_warns_on_tainted_attr
|
|
||||||
with_taint_mode(:warn) do
|
|
||||||
tpl = Liquid::Template.parse('{{ product.user_input }}')
|
|
||||||
context = Context.new('product' => ProductDrop.new)
|
|
||||||
tpl.render!(context)
|
|
||||||
assert_equal [Liquid::TaintedError], context.warnings.map(&:class)
|
|
||||||
assert_equal "variable 'product.user_input' is tainted and was not escaped", context.warnings.first.to_s(false)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_rendering_doesnt_raise_on_escaped_tainted_attr
|
|
||||||
with_taint_mode(:error) do
|
|
||||||
tpl = Liquid::Template.parse('{{ product.user_input | escape }}')
|
|
||||||
tpl.render!('product' => ProductDrop.new)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_drop_does_only_respond_to_whitelisted_methods
|
def test_drop_does_only_respond_to_whitelisted_methods
|
||||||
assert_equal "", Liquid::Template.parse("{{ product.inspect }}").render!('product' => ProductDrop.new)
|
assert_equal("", Liquid::Template.parse("{{ product.inspect }}").render!('product' => ProductDrop.new))
|
||||||
assert_equal "", Liquid::Template.parse("{{ product.pretty_inspect }}").render!('product' => ProductDrop.new)
|
assert_equal("", Liquid::Template.parse("{{ product.pretty_inspect }}").render!('product' => ProductDrop.new))
|
||||||
assert_equal "", Liquid::Template.parse("{{ product.whatever }}").render!('product' => ProductDrop.new)
|
assert_equal("", Liquid::Template.parse("{{ product.whatever }}").render!('product' => ProductDrop.new))
|
||||||
assert_equal "", Liquid::Template.parse('{{ product | map: "inspect" }}').render!('product' => ProductDrop.new)
|
assert_equal("", Liquid::Template.parse('{{ product | map: "inspect" }}').render!('product' => ProductDrop.new))
|
||||||
assert_equal "", Liquid::Template.parse('{{ product | map: "pretty_inspect" }}').render!('product' => ProductDrop.new)
|
assert_equal("", Liquid::Template.parse('{{ product | map: "pretty_inspect" }}').render!('product' => ProductDrop.new))
|
||||||
assert_equal "", Liquid::Template.parse('{{ product | map: "whatever" }}').render!('product' => ProductDrop.new)
|
assert_equal("", Liquid::Template.parse('{{ product | map: "whatever" }}').render!('product' => ProductDrop.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_drops_respond_to_to_liquid
|
def test_drops_respond_to_to_liquid
|
||||||
assert_equal "text1", Liquid::Template.parse("{{ product.to_liquid.texts.text }}").render!('product' => ProductDrop.new)
|
assert_equal("text1", Liquid::Template.parse("{{ product.to_liquid.texts.text }}").render!('product' => ProductDrop.new))
|
||||||
assert_equal "text1", Liquid::Template.parse('{{ product | map: "to_liquid" | map: "texts" | map: "text" }}').render!('product' => ProductDrop.new)
|
assert_equal("text1", Liquid::Template.parse('{{ product | map: "to_liquid" | map: "texts" | map: "text" }}').render!('product' => ProductDrop.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_text_drop
|
def test_text_drop
|
||||||
output = Liquid::Template.parse(' {{ product.texts.text }} ').render!('product' => ProductDrop.new)
|
output = Liquid::Template.parse(' {{ product.texts.text }} ').render!('product' => ProductDrop.new)
|
||||||
assert_equal ' text1 ', output
|
assert_equal(' text1 ', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_catchall_unknown_method
|
def test_catchall_unknown_method
|
||||||
output = Liquid::Template.parse(' {{ product.catchall.unknown }} ').render!('product' => ProductDrop.new)
|
output = Liquid::Template.parse(' {{ product.catchall.unknown }} ').render!('product' => ProductDrop.new)
|
||||||
assert_equal ' catchall_method: unknown ', output
|
assert_equal(' catchall_method: unknown ', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_catchall_integer_argument_drop
|
def test_catchall_integer_argument_drop
|
||||||
output = Liquid::Template.parse(' {{ product.catchall[8] }} ').render!('product' => ProductDrop.new)
|
output = Liquid::Template.parse(' {{ product.catchall[8] }} ').render!('product' => ProductDrop.new)
|
||||||
assert_equal ' catchall_method: 8 ', output
|
assert_equal(' catchall_method: 8 ', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_text_array_drop
|
def test_text_array_drop
|
||||||
output = Liquid::Template.parse('{% for text in product.texts.array %} {{text}} {% endfor %}').render!('product' => ProductDrop.new)
|
output = Liquid::Template.parse('{% for text in product.texts.array %} {{text}} {% endfor %}').render!('product' => ProductDrop.new)
|
||||||
assert_equal ' text1 text2 ', output
|
assert_equal(' text1 text2 ', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_context_drop
|
def test_context_drop
|
||||||
output = Liquid::Template.parse(' {{ context.bar }} ').render!('context' => ContextDrop.new, 'bar' => "carrot")
|
output = Liquid::Template.parse(' {{ context.bar }} ').render!('context' => ContextDrop.new, 'bar' => "carrot")
|
||||||
assert_equal ' carrot ', output
|
assert_equal(' carrot ', output)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_context_drop_array_with_map
|
||||||
|
output = Liquid::Template.parse(' {{ contexts | map: "bar" }} ').render!('contexts' => [ContextDrop.new, ContextDrop.new], 'bar' => "carrot")
|
||||||
|
assert_equal(' carrotcarrot ', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nested_context_drop
|
def test_nested_context_drop
|
||||||
output = Liquid::Template.parse(' {{ product.context.foo }} ').render!('product' => ProductDrop.new, 'foo' => "monkey")
|
output = Liquid::Template.parse(' {{ product.context.foo }} ').render!('product' => ProductDrop.new, 'foo' => "monkey")
|
||||||
assert_equal ' monkey ', output
|
assert_equal(' monkey ', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_protected
|
def test_protected
|
||||||
output = Liquid::Template.parse(' {{ product.callmenot }} ').render!('product' => ProductDrop.new)
|
output = Liquid::Template.parse(' {{ product.callmenot }} ').render!('product' => ProductDrop.new)
|
||||||
assert_equal ' ', output
|
assert_equal(' ', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_object_methods_not_allowed
|
def test_object_methods_not_allowed
|
||||||
[:dup, :clone, :singleton_class, :eval, :class_eval, :inspect].each do |method|
|
[:dup, :clone, :singleton_class, :eval, :class_eval, :inspect].each do |method|
|
||||||
output = Liquid::Template.parse(" {{ product.#{method} }} ").render!('product' => ProductDrop.new)
|
output = Liquid::Template.parse(" {{ product.#{method} }} ").render!('product' => ProductDrop.new)
|
||||||
assert_equal ' ', output
|
assert_equal(' ', output)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_scope
|
def test_scope
|
||||||
assert_equal '1', Liquid::Template.parse('{{ context.scopes }}').render!('context' => ContextDrop.new)
|
assert_equal('1', Liquid::Template.parse('{{ context.scopes }}').render!('context' => ContextDrop.new))
|
||||||
assert_equal '2', Liquid::Template.parse('{%for i in dummy%}{{ context.scopes }}{%endfor%}').render!('context' => ContextDrop.new, 'dummy' => [1])
|
assert_equal('2', Liquid::Template.parse('{%for i in dummy%}{{ context.scopes }}{%endfor%}').render!('context' => ContextDrop.new, 'dummy' => [1]))
|
||||||
assert_equal '3', Liquid::Template.parse('{%for i in dummy%}{%for i in dummy%}{{ context.scopes }}{%endfor%}{%endfor%}').render!('context' => ContextDrop.new, 'dummy' => [1])
|
assert_equal('3', Liquid::Template.parse('{%for i in dummy%}{%for i in dummy%}{{ context.scopes }}{%endfor%}{%endfor%}').render!('context' => ContextDrop.new, 'dummy' => [1]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_scope_though_proc
|
def test_scope_though_proc
|
||||||
assert_equal '1', Liquid::Template.parse('{{ s }}').render!('context' => ContextDrop.new, 's' => proc { |c| c['context.scopes'] })
|
assert_equal('1', Liquid::Template.parse('{{ s }}').render!('context' => ContextDrop.new, 's' => proc { |c| c['context.scopes'] }))
|
||||||
assert_equal '2', Liquid::Template.parse('{%for i in dummy%}{{ s }}{%endfor%}').render!('context' => ContextDrop.new, 's' => proc { |c| c['context.scopes'] }, 'dummy' => [1])
|
assert_equal('2', Liquid::Template.parse('{%for i in dummy%}{{ s }}{%endfor%}').render!('context' => ContextDrop.new, 's' => proc { |c| c['context.scopes'] }, 'dummy' => [1]))
|
||||||
assert_equal '3', Liquid::Template.parse('{%for i in dummy%}{%for i in dummy%}{{ s }}{%endfor%}{%endfor%}').render!('context' => ContextDrop.new, 's' => proc { |c| c['context.scopes'] }, 'dummy' => [1])
|
assert_equal('3', Liquid::Template.parse('{%for i in dummy%}{%for i in dummy%}{{ s }}{%endfor%}{%endfor%}').render!('context' => ContextDrop.new, 's' => proc { |c| c['context.scopes'] }, 'dummy' => [1]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_scope_with_assigns
|
def test_scope_with_assigns
|
||||||
assert_equal 'variable', Liquid::Template.parse('{% assign a = "variable"%}{{a}}').render!('context' => ContextDrop.new)
|
assert_equal('variable', Liquid::Template.parse('{% assign a = "variable"%}{{a}}').render!('context' => ContextDrop.new))
|
||||||
assert_equal 'variable', Liquid::Template.parse('{% assign a = "variable"%}{%for i in dummy%}{{a}}{%endfor%}').render!('context' => ContextDrop.new, 'dummy' => [1])
|
assert_equal('variable', Liquid::Template.parse('{% assign a = "variable"%}{%for i in dummy%}{{a}}{%endfor%}').render!('context' => ContextDrop.new, 'dummy' => [1]))
|
||||||
assert_equal 'test', Liquid::Template.parse('{% assign header_gif = "test"%}{{header_gif}}').render!('context' => ContextDrop.new)
|
assert_equal('test', Liquid::Template.parse('{% assign header_gif = "test"%}{{header_gif}}').render!('context' => ContextDrop.new))
|
||||||
assert_equal 'test', Liquid::Template.parse("{% assign header_gif = 'test'%}{{header_gif}}").render!('context' => ContextDrop.new)
|
assert_equal('test', Liquid::Template.parse("{% assign header_gif = 'test'%}{{header_gif}}").render!('context' => ContextDrop.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_scope_from_tags
|
def test_scope_from_tags
|
||||||
assert_equal '1', Liquid::Template.parse('{% for i in context.scopes_as_array %}{{i}}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1])
|
assert_equal('1', Liquid::Template.parse('{% for i in context.scopes_as_array %}{{i}}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1]))
|
||||||
assert_equal '12', Liquid::Template.parse('{%for a in dummy%}{% for i in context.scopes_as_array %}{{i}}{% endfor %}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1])
|
assert_equal('12', Liquid::Template.parse('{%for a in dummy%}{% for i in context.scopes_as_array %}{{i}}{% endfor %}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1]))
|
||||||
assert_equal '123', Liquid::Template.parse('{%for a in dummy%}{%for a in dummy%}{% for i in context.scopes_as_array %}{{i}}{% endfor %}{% endfor %}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1])
|
assert_equal('123', Liquid::Template.parse('{%for a in dummy%}{%for a in dummy%}{% for i in context.scopes_as_array %}{{i}}{% endfor %}{% endfor %}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_access_context_from_drop
|
def test_access_context_from_drop
|
||||||
assert_equal '123', Liquid::Template.parse('{%for a in dummy%}{{ context.loop_pos }}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1, 2, 3])
|
assert_equal('123', Liquid::Template.parse('{%for a in dummy%}{{ context.loop_pos }}{% endfor %}').render!('context' => ContextDrop.new, 'dummy' => [1, 2, 3]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_enumerable_drop
|
def test_enumerable_drop
|
||||||
assert_equal '123', Liquid::Template.parse('{% for c in collection %}{{c}}{% endfor %}').render!('collection' => EnumerableDrop.new)
|
assert_equal('123', Liquid::Template.parse('{% for c in collection %}{{c}}{% endfor %}').render!('collection' => EnumerableDrop.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_enumerable_drop_size
|
def test_enumerable_drop_size
|
||||||
assert_equal '3', Liquid::Template.parse('{{collection.size}}').render!('collection' => EnumerableDrop.new)
|
assert_equal('3', Liquid::Template.parse('{{collection.size}}').render!('collection' => EnumerableDrop.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_enumerable_drop_will_invoke_liquid_method_missing_for_clashing_method_names
|
def test_enumerable_drop_will_invoke_liquid_method_missing_for_clashing_method_names
|
||||||
["select", "each", "map", "cycle"].each do |method|
|
["select", "each", "map", "cycle"].each do |method|
|
||||||
assert_equal method.to_s, Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => EnumerableDrop.new)
|
assert_equal(method.to_s, Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => EnumerableDrop.new))
|
||||||
assert_equal method.to_s, Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => EnumerableDrop.new)
|
assert_equal(method.to_s, Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => EnumerableDrop.new))
|
||||||
assert_equal method.to_s, Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => RealEnumerableDrop.new)
|
assert_equal(method.to_s, Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => RealEnumerableDrop.new))
|
||||||
assert_equal method.to_s, Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => RealEnumerableDrop.new)
|
assert_equal(method.to_s, Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => RealEnumerableDrop.new))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_some_enumerable_methods_still_get_invoked
|
def test_some_enumerable_methods_still_get_invoked
|
||||||
[:count, :max].each do |method|
|
[:count, :max].each do |method|
|
||||||
assert_equal "3", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => RealEnumerableDrop.new)
|
assert_equal("3", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => RealEnumerableDrop.new))
|
||||||
assert_equal "3", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => RealEnumerableDrop.new)
|
assert_equal("3", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => RealEnumerableDrop.new))
|
||||||
assert_equal "3", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => EnumerableDrop.new)
|
assert_equal("3", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => EnumerableDrop.new))
|
||||||
assert_equal "3", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => EnumerableDrop.new)
|
assert_equal("3", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => EnumerableDrop.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_equal "yes", Liquid::Template.parse("{% if collection contains 3 %}yes{% endif %}").render!('collection' => RealEnumerableDrop.new)
|
assert_equal("yes", Liquid::Template.parse("{% if collection contains 3 %}yes{% endif %}").render!('collection' => RealEnumerableDrop.new))
|
||||||
|
|
||||||
[:min, :first].each do |method|
|
[:min, :first].each do |method|
|
||||||
assert_equal "1", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => RealEnumerableDrop.new)
|
assert_equal("1", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => RealEnumerableDrop.new))
|
||||||
assert_equal "1", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => RealEnumerableDrop.new)
|
assert_equal("1", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => RealEnumerableDrop.new))
|
||||||
assert_equal "1", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => EnumerableDrop.new)
|
assert_equal("1", Liquid::Template.parse("{{collection.#{method}}}").render!('collection' => EnumerableDrop.new))
|
||||||
assert_equal "1", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => EnumerableDrop.new)
|
assert_equal("1", Liquid::Template.parse("{{collection[\"#{method}\"]}}").render!('collection' => EnumerableDrop.new))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_empty_string_value_access
|
def test_empty_string_value_access
|
||||||
assert_equal '', Liquid::Template.parse('{{ product[value] }}').render!('product' => ProductDrop.new, 'value' => '')
|
assert_equal('', Liquid::Template.parse('{{ product[value] }}').render!('product' => ProductDrop.new, 'value' => ''))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nil_value_access
|
def test_nil_value_access
|
||||||
assert_equal '', Liquid::Template.parse('{{ product[value] }}').render!('product' => ProductDrop.new, 'value' => nil)
|
assert_equal('', Liquid::Template.parse('{{ product[value] }}').render!('product' => ProductDrop.new, 'value' => nil))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_default_to_s_on_drops
|
def test_default_to_s_on_drops
|
||||||
assert_equal 'ProductDrop', Liquid::Template.parse("{{ product }}").render!('product' => ProductDrop.new)
|
assert_equal('ProductDrop', Liquid::Template.parse("{{ product }}").render!('product' => ProductDrop.new))
|
||||||
assert_equal 'EnumerableDrop', Liquid::Template.parse('{{ collection }}').render!('collection' => EnumerableDrop.new)
|
assert_equal('EnumerableDrop', Liquid::Template.parse('{{ collection }}').render!('collection' => EnumerableDrop.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_invokable_methods
|
def test_invokable_methods
|
||||||
assert_equal %w(to_liquid catchall user_input context texts).to_set, ProductDrop.invokable_methods
|
assert_equal(%w(to_liquid catchall context texts).to_set, ProductDrop.invokable_methods)
|
||||||
assert_equal %w(to_liquid scopes_as_array loop_pos scopes).to_set, ContextDrop.invokable_methods
|
assert_equal(%w(to_liquid scopes_as_array loop_pos scopes).to_set, ContextDrop.invokable_methods)
|
||||||
assert_equal %w(to_liquid size max min first count).to_set, EnumerableDrop.invokable_methods
|
assert_equal(%w(to_liquid size max min first count).to_set, EnumerableDrop.invokable_methods)
|
||||||
assert_equal %w(to_liquid max min sort count first).to_set, RealEnumerableDrop.invokable_methods
|
assert_equal(%w(to_liquid max min sort count first).to_set, RealEnumerableDrop.invokable_methods)
|
||||||
end
|
end
|
||||||
end # DropsTest
|
end # DropsTest
|
||||||
|
|||||||
@@ -35,31 +35,31 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
TEXT
|
TEXT
|
||||||
|
|
||||||
output = Liquid::Template.parse(template, line_numbers: true).render('errors' => ErrorDrop.new)
|
output = Liquid::Template.parse(template, line_numbers: true).render('errors' => ErrorDrop.new)
|
||||||
assert_equal expected, output
|
assert_equal(expected, output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_standard_error
|
def test_standard_error
|
||||||
template = Liquid::Template.parse(' {{ errors.standard_error }} ')
|
template = Liquid::Template.parse(' {{ errors.standard_error }} ')
|
||||||
assert_equal ' Liquid error: standard error ', template.render('errors' => ErrorDrop.new)
|
assert_equal(' Liquid error: standard error ', template.render('errors' => ErrorDrop.new))
|
||||||
|
|
||||||
assert_equal 1, template.errors.size
|
assert_equal(1, template.errors.size)
|
||||||
assert_equal StandardError, template.errors.first.class
|
assert_equal(StandardError, template.errors.first.class)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_syntax
|
def test_syntax
|
||||||
template = Liquid::Template.parse(' {{ errors.syntax_error }} ')
|
template = Liquid::Template.parse(' {{ errors.syntax_error }} ')
|
||||||
assert_equal ' Liquid syntax error: syntax error ', template.render('errors' => ErrorDrop.new)
|
assert_equal(' Liquid syntax error: syntax error ', template.render('errors' => ErrorDrop.new))
|
||||||
|
|
||||||
assert_equal 1, template.errors.size
|
assert_equal(1, template.errors.size)
|
||||||
assert_equal SyntaxError, template.errors.first.class
|
assert_equal(SyntaxError, template.errors.first.class)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_argument
|
def test_argument
|
||||||
template = Liquid::Template.parse(' {{ errors.argument_error }} ')
|
template = Liquid::Template.parse(' {{ errors.argument_error }} ')
|
||||||
assert_equal ' Liquid error: argument error ', template.render('errors' => ErrorDrop.new)
|
assert_equal(' Liquid error: argument error ', template.render('errors' => ErrorDrop.new))
|
||||||
|
|
||||||
assert_equal 1, template.errors.size
|
assert_equal(1, template.errors.size)
|
||||||
assert_equal ArgumentError, template.errors.first.class
|
assert_equal(ArgumentError, template.errors.first.class)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_missing_endtag_parse_time_error
|
def test_missing_endtag_parse_time_error
|
||||||
@@ -78,9 +78,9 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
|
|
||||||
def test_lax_unrecognized_operator
|
def test_lax_unrecognized_operator
|
||||||
template = Liquid::Template.parse(' {% if 1 =! 2 %}ok{% endif %} ', error_mode: :lax)
|
template = Liquid::Template.parse(' {% if 1 =! 2 %}ok{% endif %} ', error_mode: :lax)
|
||||||
assert_equal ' Liquid error: Unknown operator =! ', template.render
|
assert_equal(' Liquid error: Unknown operator =! ', template.render)
|
||||||
assert_equal 1, template.errors.size
|
assert_equal(1, template.errors.size)
|
||||||
assert_equal Liquid::ArgumentError, template.errors.first.class
|
assert_equal(Liquid::ArgumentError, template.errors.first.class)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_with_line_numbers_adds_numbers_to_parser_errors
|
def test_with_line_numbers_adds_numbers_to_parser_errors
|
||||||
@@ -124,8 +124,8 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
error_mode: :warn,
|
error_mode: :warn,
|
||||||
line_numbers: true)
|
line_numbers: true)
|
||||||
|
|
||||||
assert_equal ['Liquid syntax error (line 4): Unexpected character = in "1 =! 2"'],
|
assert_equal(['Liquid syntax error (line 4): Unexpected character = in "1 =! 2"'],
|
||||||
template.warnings.map(&:message)
|
template.warnings.map(&:message))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_parsing_strict_with_line_numbers_adds_numbers_to_lexer_errors
|
def test_parsing_strict_with_line_numbers_adds_numbers_to_lexer_errors
|
||||||
@@ -141,7 +141,7 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
line_numbers: true)
|
line_numbers: true)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_equal 'Liquid syntax error (line 4): Unexpected character = in "1 =! 2"', err.message
|
assert_equal('Liquid syntax error (line 4): Unexpected character = in "1 =! 2"', err.message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_syntax_errors_in_nested_blocks_have_correct_line_number
|
def test_syntax_errors_in_nested_blocks_have_correct_line_number
|
||||||
@@ -158,42 +158,42 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
line_numbers: true)
|
line_numbers: true)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_equal "Liquid syntax error (line 5): Unknown tag 'foo'", err.message
|
assert_equal("Liquid syntax error (line 5): Unknown tag 'foo'", err.message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strict_error_messages
|
def test_strict_error_messages
|
||||||
err = assert_raises(SyntaxError) do
|
err = assert_raises(SyntaxError) do
|
||||||
Liquid::Template.parse(' {% if 1 =! 2 %}ok{% endif %} ', error_mode: :strict)
|
Liquid::Template.parse(' {% if 1 =! 2 %}ok{% endif %} ', error_mode: :strict)
|
||||||
end
|
end
|
||||||
assert_equal 'Liquid syntax error: Unexpected character = in "1 =! 2"', err.message
|
assert_equal('Liquid syntax error: Unexpected character = in "1 =! 2"', err.message)
|
||||||
|
|
||||||
err = assert_raises(SyntaxError) do
|
err = assert_raises(SyntaxError) do
|
||||||
Liquid::Template.parse('{{%%%}}', error_mode: :strict)
|
Liquid::Template.parse('{{%%%}}', error_mode: :strict)
|
||||||
end
|
end
|
||||||
assert_equal 'Liquid syntax error: Unexpected character % in "{{%%%}}"', err.message
|
assert_equal('Liquid syntax error: Unexpected character % in "{{%%%}}"', err.message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_warnings
|
def test_warnings
|
||||||
template = Liquid::Template.parse('{% if ~~~ %}{{%%%}}{% else %}{{ hello. }}{% endif %}', error_mode: :warn)
|
template = Liquid::Template.parse('{% if ~~~ %}{{%%%}}{% else %}{{ hello. }}{% endif %}', error_mode: :warn)
|
||||||
assert_equal 3, template.warnings.size
|
assert_equal(3, template.warnings.size)
|
||||||
assert_equal 'Unexpected character ~ in "~~~"', template.warnings[0].to_s(false)
|
assert_equal('Unexpected character ~ in "~~~"', template.warnings[0].to_s(false))
|
||||||
assert_equal 'Unexpected character % in "{{%%%}}"', template.warnings[1].to_s(false)
|
assert_equal('Unexpected character % in "{{%%%}}"', template.warnings[1].to_s(false))
|
||||||
assert_equal 'Expected id but found end_of_string in "{{ hello. }}"', template.warnings[2].to_s(false)
|
assert_equal('Expected id but found end_of_string in "{{ hello. }}"', template.warnings[2].to_s(false))
|
||||||
assert_equal '', template.render
|
assert_equal('', template.render)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_warning_line_numbers
|
def test_warning_line_numbers
|
||||||
template = Liquid::Template.parse("{% if ~~~ %}\n{{%%%}}{% else %}\n{{ hello. }}{% endif %}", error_mode: :warn, line_numbers: true)
|
template = Liquid::Template.parse("{% if ~~~ %}\n{{%%%}}{% else %}\n{{ hello. }}{% endif %}", error_mode: :warn, line_numbers: true)
|
||||||
assert_equal 'Liquid syntax error (line 1): Unexpected character ~ in "~~~"', template.warnings[0].message
|
assert_equal('Liquid syntax error (line 1): Unexpected character ~ in "~~~"', template.warnings[0].message)
|
||||||
assert_equal 'Liquid syntax error (line 2): Unexpected character % in "{{%%%}}"', template.warnings[1].message
|
assert_equal('Liquid syntax error (line 2): Unexpected character % in "{{%%%}}"', template.warnings[1].message)
|
||||||
assert_equal 'Liquid syntax error (line 3): Expected id but found end_of_string in "{{ hello. }}"', template.warnings[2].message
|
assert_equal('Liquid syntax error (line 3): Expected id but found end_of_string in "{{ hello. }}"', template.warnings[2].message)
|
||||||
assert_equal 3, template.warnings.size
|
assert_equal(3, template.warnings.size)
|
||||||
assert_equal [1, 2, 3], template.warnings.map(&:line_number)
|
assert_equal([1, 2, 3], template.warnings.map(&:line_number))
|
||||||
end
|
end
|
||||||
|
|
||||||
# Liquid should not catch Exceptions that are not subclasses of StandardError, like Interrupt and NoMemoryError
|
# Liquid should not catch Exceptions that are not subclasses of StandardError, like Interrupt and NoMemoryError
|
||||||
def test_exceptions_propagate
|
def test_exceptions_propagate
|
||||||
assert_raises Exception do
|
assert_raises(Exception) do
|
||||||
template = Liquid::Template.parse('{{ errors.exception }}')
|
template = Liquid::Template.parse('{{ errors.exception }}')
|
||||||
template.render('errors' => ErrorDrop.new)
|
template.render('errors' => ErrorDrop.new)
|
||||||
end
|
end
|
||||||
@@ -204,8 +204,8 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
|
|
||||||
output = template.render('errors' => ErrorDrop.new)
|
output = template.render('errors' => ErrorDrop.new)
|
||||||
|
|
||||||
assert_equal 'This is a runtime error: Liquid error (line 1): internal', output
|
assert_equal('This is a runtime error: Liquid error (line 1): internal', output)
|
||||||
assert_equal [Liquid::InternalError], template.errors.map(&:class)
|
assert_equal([Liquid::InternalError], template.errors.map(&:class))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_setting_default_exception_renderer
|
def test_setting_default_exception_renderer
|
||||||
@@ -219,8 +219,8 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
|
|
||||||
output = template.render('errors' => ErrorDrop.new)
|
output = template.render('errors' => ErrorDrop.new)
|
||||||
|
|
||||||
assert_equal 'This is a runtime error: ', output
|
assert_equal('This is a runtime error: ', output)
|
||||||
assert_equal [Liquid::ArgumentError], template.errors.map(&:class)
|
assert_equal([Liquid::ArgumentError], template.errors.map(&:class))
|
||||||
ensure
|
ensure
|
||||||
Liquid::Template.default_exception_renderer = old_exception_renderer if old_exception_renderer
|
Liquid::Template.default_exception_renderer = old_exception_renderer if old_exception_renderer
|
||||||
end
|
end
|
||||||
@@ -235,10 +235,10 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
|
|
||||||
output = template.render({ 'errors' => ErrorDrop.new }, exception_renderer: handler)
|
output = template.render({ 'errors' => ErrorDrop.new }, exception_renderer: handler)
|
||||||
|
|
||||||
assert_equal 'This is a runtime error: runtime error', output
|
assert_equal('This is a runtime error: runtime error', output)
|
||||||
assert_equal [Liquid::InternalError], exceptions.map(&:class)
|
assert_equal([Liquid::InternalError], exceptions.map(&:class))
|
||||||
assert_equal exceptions, template.errors
|
assert_equal(exceptions, template.errors)
|
||||||
assert_equal '#<RuntimeError: runtime error>', exceptions.first.cause.inspect
|
assert_equal('#<RuntimeError: runtime error>', exceptions.first.cause.inspect)
|
||||||
end
|
end
|
||||||
|
|
||||||
class TestFileSystem
|
class TestFileSystem
|
||||||
@@ -252,12 +252,21 @@ class ErrorHandlingTest < Minitest::Test
|
|||||||
|
|
||||||
begin
|
begin
|
||||||
Liquid::Template.file_system = TestFileSystem.new
|
Liquid::Template.file_system = TestFileSystem.new
|
||||||
|
|
||||||
template = Liquid::Template.parse("Argument error:\n{% include 'product' %}", line_numbers: true)
|
template = Liquid::Template.parse("Argument error:\n{% include 'product' %}", line_numbers: true)
|
||||||
page = template.render('errors' => ErrorDrop.new)
|
page = template.render('errors' => ErrorDrop.new)
|
||||||
ensure
|
ensure
|
||||||
Liquid::Template.file_system = old_file_system
|
Liquid::Template.file_system = old_file_system
|
||||||
end
|
end
|
||||||
assert_equal "Argument error:\nLiquid error (product line 1): argument error", page
|
assert_equal("Argument error:\nLiquid error (product line 1): argument error", page)
|
||||||
assert_equal "product", template.errors.first.template_name
|
assert_equal("product", template.errors.first.template_name)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_bug_compatible_silencing_of_errors_in_blank_nodes
|
||||||
|
output = Liquid::Template.parse("{% assign x = 0 %}{% if 1 < '2' %}not blank{% assign x = 3 %}{% endif %}{{ x }}").render
|
||||||
|
assert_equal("Liquid error: comparison of Integer with String failed0", output)
|
||||||
|
|
||||||
|
output = Liquid::Template.parse("{% assign x = 0 %}{% if 1 < '2' %}{% assign x = 3 %}{% endif %}{{ x }}").render
|
||||||
|
assert_equal("0", output)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
46
test/integration/expression_test.rb
Normal file
46
test/integration/expression_test.rb
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'test_helper'
|
||||||
|
|
||||||
|
class ExpressionTest < Minitest::Test
|
||||||
|
def test_keyword_literals
|
||||||
|
assert_equal(true, parse_and_eval("true"))
|
||||||
|
assert_equal(true, parse_and_eval(" true "))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_string
|
||||||
|
assert_equal("single quoted", parse_and_eval("'single quoted'"))
|
||||||
|
assert_equal("double quoted", parse_and_eval('"double quoted"'))
|
||||||
|
assert_equal("spaced", parse_and_eval(" 'spaced' "))
|
||||||
|
assert_equal("spaced2", parse_and_eval(' "spaced2" '))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_int
|
||||||
|
assert_equal(123, parse_and_eval("123"))
|
||||||
|
assert_equal(456, parse_and_eval(" 456 "))
|
||||||
|
assert_equal(12, parse_and_eval("012"))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_float
|
||||||
|
assert_equal(1.5, parse_and_eval("1.5"))
|
||||||
|
assert_equal(2.5, parse_and_eval(" 2.5 "))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_range
|
||||||
|
assert_equal(1..2, parse_and_eval("(1..2)"))
|
||||||
|
assert_equal(3..4, parse_and_eval(" ( 3 .. 4 ) "))
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def parse_and_eval(markup, **assigns)
|
||||||
|
if Liquid::Template.error_mode == :strict
|
||||||
|
p = Liquid::Parser.new(markup)
|
||||||
|
markup = p.expression
|
||||||
|
p.consume(:end_of_string)
|
||||||
|
end
|
||||||
|
expression = Liquid::Expression.parse(markup)
|
||||||
|
context = Liquid::Context.new(assigns)
|
||||||
|
context.evaluate(expression)
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -41,13 +41,13 @@ class FiltersTest < Minitest::Test
|
|||||||
@context['var'] = 1000
|
@context['var'] = 1000
|
||||||
@context.add_filters(MoneyFilter)
|
@context.add_filters(MoneyFilter)
|
||||||
|
|
||||||
assert_equal ' 1000$ ', Template.parse("{{var | money}}").render(@context)
|
assert_equal(' 1000$ ', Template.parse("{{var | money}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_underscore_in_filter_name
|
def test_underscore_in_filter_name
|
||||||
@context['var'] = 1000
|
@context['var'] = 1000
|
||||||
@context.add_filters(MoneyFilter)
|
@context.add_filters(MoneyFilter)
|
||||||
assert_equal ' 1000$ ', Template.parse("{{var | money_with_underscore}}").render(@context)
|
assert_equal(' 1000$ ', Template.parse("{{var | money_with_underscore}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_second_filter_overwrites_first
|
def test_second_filter_overwrites_first
|
||||||
@@ -55,20 +55,20 @@ class FiltersTest < Minitest::Test
|
|||||||
@context.add_filters(MoneyFilter)
|
@context.add_filters(MoneyFilter)
|
||||||
@context.add_filters(CanadianMoneyFilter)
|
@context.add_filters(CanadianMoneyFilter)
|
||||||
|
|
||||||
assert_equal ' 1000$ CAD ', Template.parse("{{var | money}}").render(@context)
|
assert_equal(' 1000$ CAD ', Template.parse("{{var | money}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_size
|
def test_size
|
||||||
@context['var'] = 'abcd'
|
@context['var'] = 'abcd'
|
||||||
@context.add_filters(MoneyFilter)
|
@context.add_filters(MoneyFilter)
|
||||||
|
|
||||||
assert_equal '4', Template.parse("{{var | size}}").render(@context)
|
assert_equal('4', Template.parse("{{var | size}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_join
|
def test_join
|
||||||
@context['var'] = [1, 2, 3, 4]
|
@context['var'] = [1, 2, 3, 4]
|
||||||
|
|
||||||
assert_equal "1 2 3 4", Template.parse("{{var | join}}").render(@context)
|
assert_equal("1 2 3 4", Template.parse("{{var | join}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort
|
def test_sort
|
||||||
@@ -78,11 +78,11 @@ class FiltersTest < Minitest::Test
|
|||||||
@context['arrays'] = ['flower', 'are']
|
@context['arrays'] = ['flower', 'are']
|
||||||
@context['case_sensitive'] = ['sensitive', 'Expected', 'case']
|
@context['case_sensitive'] = ['sensitive', 'Expected', 'case']
|
||||||
|
|
||||||
assert_equal '1 2 3 4', Template.parse("{{numbers | sort | join}}").render(@context)
|
assert_equal('1 2 3 4', Template.parse("{{numbers | sort | join}}").render(@context))
|
||||||
assert_equal 'alphabetic as expected', Template.parse("{{words | sort | join}}").render(@context)
|
assert_equal('alphabetic as expected', Template.parse("{{words | sort | join}}").render(@context))
|
||||||
assert_equal '3', Template.parse("{{value | sort}}").render(@context)
|
assert_equal('3', Template.parse("{{value | sort}}").render(@context))
|
||||||
assert_equal 'are flower', Template.parse("{{arrays | sort | join}}").render(@context)
|
assert_equal('are flower', Template.parse("{{arrays | sort | join}}").render(@context))
|
||||||
assert_equal 'Expected case sensitive', Template.parse("{{case_sensitive | sort | join}}").render(@context)
|
assert_equal('Expected case sensitive', Template.parse("{{case_sensitive | sort | join}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_natural
|
def test_sort_natural
|
||||||
@@ -91,13 +91,13 @@ class FiltersTest < Minitest::Test
|
|||||||
@context['objects'] = [TestObject.new('A'), TestObject.new('b'), TestObject.new('C')]
|
@context['objects'] = [TestObject.new('A'), TestObject.new('b'), TestObject.new('C')]
|
||||||
|
|
||||||
# Test strings
|
# Test strings
|
||||||
assert_equal 'Assert case Insensitive', Template.parse("{{words | sort_natural | join}}").render(@context)
|
assert_equal('Assert case Insensitive', Template.parse("{{words | sort_natural | join}}").render(@context))
|
||||||
|
|
||||||
# Test hashes
|
# Test hashes
|
||||||
assert_equal 'A b C', Template.parse("{{hashes | sort_natural: 'a' | map: 'a' | join}}").render(@context)
|
assert_equal('A b C', Template.parse("{{hashes | sort_natural: 'a' | map: 'a' | join}}").render(@context))
|
||||||
|
|
||||||
# Test objects
|
# Test objects
|
||||||
assert_equal 'A b C', Template.parse("{{objects | sort_natural: 'a' | map: 'a' | join}}").render(@context)
|
assert_equal('A b C', Template.parse("{{objects | sort_natural: 'a' | map: 'a' | join}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_compact
|
def test_compact
|
||||||
@@ -106,37 +106,37 @@ class FiltersTest < Minitest::Test
|
|||||||
@context['objects'] = [TestObject.new('A'), TestObject.new(nil), TestObject.new('C')]
|
@context['objects'] = [TestObject.new('A'), TestObject.new(nil), TestObject.new('C')]
|
||||||
|
|
||||||
# Test strings
|
# Test strings
|
||||||
assert_equal 'a b c', Template.parse("{{words | compact | join}}").render(@context)
|
assert_equal('a b c', Template.parse("{{words | compact | join}}").render(@context))
|
||||||
|
|
||||||
# Test hashes
|
# Test hashes
|
||||||
assert_equal 'A C', Template.parse("{{hashes | compact: 'a' | map: 'a' | join}}").render(@context)
|
assert_equal('A C', Template.parse("{{hashes | compact: 'a' | map: 'a' | join}}").render(@context))
|
||||||
|
|
||||||
# Test objects
|
# Test objects
|
||||||
assert_equal 'A C', Template.parse("{{objects | compact: 'a' | map: 'a' | join}}").render(@context)
|
assert_equal('A C', Template.parse("{{objects | compact: 'a' | map: 'a' | join}}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strip_html
|
def test_strip_html
|
||||||
@context['var'] = "<b>bla blub</a>"
|
@context['var'] = "<b>bla blub</a>"
|
||||||
|
|
||||||
assert_equal "bla blub", Template.parse("{{ var | strip_html }}").render(@context)
|
assert_equal("bla blub", Template.parse("{{ var | strip_html }}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strip_html_ignore_comments_with_html
|
def test_strip_html_ignore_comments_with_html
|
||||||
@context['var'] = "<!-- split and some <ul> tag --><b>bla blub</a>"
|
@context['var'] = "<!-- split and some <ul> tag --><b>bla blub</a>"
|
||||||
|
|
||||||
assert_equal "bla blub", Template.parse("{{ var | strip_html }}").render(@context)
|
assert_equal("bla blub", Template.parse("{{ var | strip_html }}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_capitalize
|
def test_capitalize
|
||||||
@context['var'] = "blub"
|
@context['var'] = "blub"
|
||||||
|
|
||||||
assert_equal "Blub", Template.parse("{{ var | capitalize }}").render(@context)
|
assert_equal("Blub", Template.parse("{{ var | capitalize }}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nonexistent_filter_is_ignored
|
def test_nonexistent_filter_is_ignored
|
||||||
@context['var'] = 1000
|
@context['var'] = 1000
|
||||||
|
|
||||||
assert_equal '1000', Template.parse("{{ var | xyzzy }}").render(@context)
|
assert_equal('1000', Template.parse("{{ var | xyzzy }}").render(@context))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_filter_with_keyword_arguments
|
def test_filter_with_keyword_arguments
|
||||||
@@ -144,14 +144,23 @@ class FiltersTest < Minitest::Test
|
|||||||
@context['input'] = 'hello %{first_name}, %{last_name}'
|
@context['input'] = 'hello %{first_name}, %{last_name}'
|
||||||
@context.add_filters(SubstituteFilter)
|
@context.add_filters(SubstituteFilter)
|
||||||
output = Template.parse(%({{ input | substitute: first_name: surname, last_name: 'doe' }})).render(@context)
|
output = Template.parse(%({{ input | substitute: first_name: surname, last_name: 'doe' }})).render(@context)
|
||||||
assert_equal 'hello john, doe', output
|
assert_equal('hello john, doe', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_override_object_method_in_filter
|
def test_override_object_method_in_filter
|
||||||
assert_equal "tap overridden", Template.parse("{{var | tap}}").render!({ 'var' => 1000 }, filters: [OverrideObjectMethodFilter])
|
assert_equal("tap overridden", Template.parse("{{var | tap}}").render!({ 'var' => 1000 }, filters: [OverrideObjectMethodFilter]))
|
||||||
|
|
||||||
# tap still treated as a non-existent filter
|
# tap still treated as a non-existent filter
|
||||||
assert_equal "1000", Template.parse("{{var | tap}}").render!('var' => 1000)
|
assert_equal("1000", Template.parse("{{var | tap}}").render!('var' => 1000))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_liquid_argument_error
|
||||||
|
source = "{{ '' | size: 'too many args' }}"
|
||||||
|
exc = assert_raises(Liquid::ArgumentError) do
|
||||||
|
Template.parse(source).render!
|
||||||
|
end
|
||||||
|
assert_match(/\ALiquid error: wrong number of arguments /, exc.message)
|
||||||
|
assert_equal(exc.message, Template.parse(source).render)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -160,15 +169,15 @@ class FiltersInTemplate < Minitest::Test
|
|||||||
|
|
||||||
def test_local_global
|
def test_local_global
|
||||||
with_global_filter(MoneyFilter) do
|
with_global_filter(MoneyFilter) do
|
||||||
assert_equal " 1000$ ", Template.parse("{{1000 | money}}").render!(nil, nil)
|
assert_equal(" 1000$ ", Template.parse("{{1000 | money}}").render!(nil, nil))
|
||||||
assert_equal " 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, filters: CanadianMoneyFilter)
|
assert_equal(" 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, filters: CanadianMoneyFilter))
|
||||||
assert_equal " 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, filters: [CanadianMoneyFilter])
|
assert_equal(" 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, filters: [CanadianMoneyFilter]))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_local_filter_with_deprecated_syntax
|
def test_local_filter_with_deprecated_syntax
|
||||||
assert_equal " 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, CanadianMoneyFilter)
|
assert_equal(" 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, CanadianMoneyFilter))
|
||||||
assert_equal " 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, [CanadianMoneyFilter])
|
assert_equal(" 1000$ CAD ", Template.parse("{{1000 | money}}").render!(nil, [CanadianMoneyFilter]))
|
||||||
end
|
end
|
||||||
end # FiltersTest
|
end # FiltersTest
|
||||||
|
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ class HashOrderingTest < Minitest::Test
|
|||||||
|
|
||||||
def test_global_register_order
|
def test_global_register_order
|
||||||
with_global_filter(MoneyFilter, CanadianMoneyFilter) do
|
with_global_filter(MoneyFilter, CanadianMoneyFilter) do
|
||||||
assert_equal " 1000$ CAD ", Template.parse("{{1000 | money}}").render(nil, nil)
|
assert_equal(" 1000$ CAD ", Template.parse("{{1000 | money}}").render(nil, nil))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -42,84 +42,84 @@ class OutputTest < Minitest::Test
|
|||||||
text = %( {{best_cars}} )
|
text = %( {{best_cars}} )
|
||||||
|
|
||||||
expected = %( bmw )
|
expected = %( bmw )
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns)
|
assert_equal(expected, Template.parse(text).render!(@assigns))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_traversing_with_two_brackets
|
def test_variable_traversing_with_two_brackets
|
||||||
text = %({{ site.data.menu[include.menu][include.locale] }})
|
text = %({{ site.data.menu[include.menu][include.locale] }})
|
||||||
assert_equal "it works!", Template.parse(text).render!(
|
assert_equal("it works!", Template.parse(text).render!(
|
||||||
"site" => { "data" => { "menu" => { "foo" => { "bar" => "it works!" } } } },
|
"site" => { "data" => { "menu" => { "foo" => { "bar" => "it works!" } } } },
|
||||||
"include" => { "menu" => "foo", "locale" => "bar" }
|
"include" => { "menu" => "foo", "locale" => "bar" }
|
||||||
)
|
))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_traversing
|
def test_variable_traversing
|
||||||
text = %( {{car.bmw}} {{car.gm}} {{car.bmw}} )
|
text = %( {{car.bmw}} {{car.gm}} {{car.bmw}} )
|
||||||
|
|
||||||
expected = %( good bad good )
|
expected = %( good bad good )
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns)
|
assert_equal(expected, Template.parse(text).render!(@assigns))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_piping
|
def test_variable_piping
|
||||||
text = %( {{ car.gm | make_funny }} )
|
text = %( {{ car.gm | make_funny }} )
|
||||||
expected = %( LOL )
|
expected = %( LOL )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_piping_with_input
|
def test_variable_piping_with_input
|
||||||
text = %( {{ car.gm | cite_funny }} )
|
text = %( {{ car.gm | cite_funny }} )
|
||||||
expected = %( LOL: bad )
|
expected = %( LOL: bad )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_piping_with_args
|
def test_variable_piping_with_args
|
||||||
text = %! {{ car.gm | add_smiley : ':-(' }} !
|
text = %! {{ car.gm | add_smiley : ':-(' }} !
|
||||||
expected = %| bad :-( |
|
expected = %| bad :-( |
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_piping_with_no_args
|
def test_variable_piping_with_no_args
|
||||||
text = %( {{ car.gm | add_smiley }} )
|
text = %( {{ car.gm | add_smiley }} )
|
||||||
expected = %| bad :-) |
|
expected = %| bad :-) |
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_multiple_variable_piping_with_args
|
def test_multiple_variable_piping_with_args
|
||||||
text = %! {{ car.gm | add_smiley : ':-(' | add_smiley : ':-('}} !
|
text = %! {{ car.gm | add_smiley : ':-(' | add_smiley : ':-('}} !
|
||||||
expected = %| bad :-( :-( |
|
expected = %| bad :-( :-( |
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_piping_with_multiple_args
|
def test_variable_piping_with_multiple_args
|
||||||
text = %( {{ car.gm | add_tag : 'span', 'bar'}} )
|
text = %( {{ car.gm | add_tag : 'span', 'bar'}} )
|
||||||
expected = %( <span id="bar">bad</span> )
|
expected = %( <span id="bar">bad</span> )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_piping_with_variable_args
|
def test_variable_piping_with_variable_args
|
||||||
text = %( {{ car.gm | add_tag : 'span', car.bmw}} )
|
text = %( {{ car.gm | add_tag : 'span', car.bmw}} )
|
||||||
expected = %( <span id="good">bad</span> )
|
expected = %( <span id="good">bad</span> )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_multiple_pipings
|
def test_multiple_pipings
|
||||||
text = %( {{ best_cars | cite_funny | paragraph }} )
|
text = %( {{ best_cars | cite_funny | paragraph }} )
|
||||||
expected = %( <p>LOL: bmw</p> )
|
expected = %( <p>LOL: bmw</p> )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_link_to
|
def test_link_to
|
||||||
text = %( {{ 'Typo' | link_to: 'http://typo.leetsoft.com' }} )
|
text = %( {{ 'Typo' | link_to: 'http://typo.leetsoft.com' }} )
|
||||||
expected = %( <a href="http://typo.leetsoft.com">Typo</a> )
|
expected = %( <a href="http://typo.leetsoft.com">Typo</a> )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter])
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: [FunnyFilter]))
|
||||||
end
|
end
|
||||||
end # OutputTest
|
end # OutputTest
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ class ParsingQuirksTest < Minitest::Test
|
|||||||
|
|
||||||
def test_parsing_css
|
def test_parsing_css
|
||||||
text = " div { font-weight: bold; } "
|
text = " div { font-weight: bold; } "
|
||||||
assert_equal text, Template.parse(text).render!
|
assert_equal(text, Template.parse(text).render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_raise_on_single_close_bracet
|
def test_raise_on_single_close_bracet
|
||||||
@@ -29,10 +29,10 @@ class ParsingQuirksTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_error_on_empty_filter
|
def test_error_on_empty_filter
|
||||||
assert Template.parse("{{test}}")
|
assert(Template.parse("{{test}}"))
|
||||||
|
|
||||||
with_error_mode(:lax) do
|
with_error_mode(:lax) do
|
||||||
assert Template.parse("{{|test}}")
|
assert(Template.parse("{{|test}}"))
|
||||||
end
|
end
|
||||||
|
|
||||||
with_error_mode(:strict) do
|
with_error_mode(:strict) do
|
||||||
@@ -64,9 +64,9 @@ class ParsingQuirksTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_no_error_on_lax_empty_filter
|
def test_no_error_on_lax_empty_filter
|
||||||
assert Template.parse("{{test |a|b|}}", error_mode: :lax)
|
assert(Template.parse("{{test |a|b|}}", error_mode: :lax))
|
||||||
assert Template.parse("{{test}}", error_mode: :lax)
|
assert(Template.parse("{{test}}", error_mode: :lax))
|
||||||
assert Template.parse("{{|test|}}", error_mode: :lax)
|
assert(Template.parse("{{|test|}}", error_mode: :lax))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_meaningless_parens_lax
|
def test_meaningless_parens_lax
|
||||||
@@ -118,6 +118,16 @@ class ParsingQuirksTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_blank_variable_markup
|
||||||
|
assert_template_result('', "{{}}")
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_lookup_on_var_with_literal_name
|
||||||
|
assigns = { "blank" => { "x" => "result" } }
|
||||||
|
assert_template_result('result', "{{ blank.x }}", assigns)
|
||||||
|
assert_template_result('result', "{{ blank['x'] }}", assigns)
|
||||||
|
end
|
||||||
|
|
||||||
def test_contains_in_id
|
def test_contains_in_id
|
||||||
assert_template_result(' YES ', '{% if containsallshipments == true %} YES {% endif %}', 'containsallshipments' => true)
|
assert_template_result(' YES ', '{% if containsallshipments == true %} YES {% endif %}', 'containsallshipments' => true)
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
require 'test_helper'
|
require 'test_helper'
|
||||||
|
|
||||||
class RenderProfilingTest < Minitest::Test
|
class ProfilerTest < Minitest::Test
|
||||||
include Liquid
|
include Liquid
|
||||||
|
|
||||||
class ProfilingFileSystem
|
class ProfilingFileSystem
|
||||||
@@ -19,35 +19,35 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
t = Template.parse("{{ 'a string' | upcase }}")
|
t = Template.parse("{{ 'a string' | upcase }}")
|
||||||
t.render!
|
t.render!
|
||||||
|
|
||||||
assert_nil t.profiler
|
assert_nil(t.profiler)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_parse_makes_available_simple_profiling
|
def test_parse_makes_available_simple_profiling
|
||||||
t = Template.parse("{{ 'a string' | upcase }}", profile: true)
|
t = Template.parse("{{ 'a string' | upcase }}", profile: true)
|
||||||
t.render!
|
t.render!
|
||||||
|
|
||||||
assert_equal 1, t.profiler.length
|
assert_equal(1, t.profiler.length)
|
||||||
|
|
||||||
node = t.profiler[0]
|
node = t.profiler[0]
|
||||||
assert_equal " 'a string' | upcase ", node.code
|
assert_equal(" 'a string' | upcase ", node.code)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_ignores_raw_strings_when_profiling
|
def test_render_ignores_raw_strings_when_profiling
|
||||||
t = Template.parse("This is raw string\nstuff\nNewline", profile: true)
|
t = Template.parse("This is raw string\nstuff\nNewline", profile: true)
|
||||||
t.render!
|
t.render!
|
||||||
|
|
||||||
assert_equal 0, t.profiler.length
|
assert_equal(0, t.profiler.length)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_includes_line_numbers_of_liquid_nodes
|
def test_profiling_includes_line_numbers_of_liquid_nodes
|
||||||
t = Template.parse("{{ 'a string' | upcase }}\n{% increment test %}", profile: true)
|
t = Template.parse("{{ 'a string' | upcase }}\n{% increment test %}", profile: true)
|
||||||
t.render!
|
t.render!
|
||||||
assert_equal 2, t.profiler.length
|
assert_equal(2, t.profiler.length)
|
||||||
|
|
||||||
# {{ 'a string' | upcase }}
|
# {{ 'a string' | upcase }}
|
||||||
assert_equal 1, t.profiler[0].line_number
|
assert_equal(1, t.profiler[0].line_number)
|
||||||
# {{ increment test }}
|
# {{ increment test }}
|
||||||
assert_equal 2, t.profiler[1].line_number
|
assert_equal(2, t.profiler[1].line_number)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_includes_line_numbers_of_included_partials
|
def test_profiling_includes_line_numbers_of_included_partials
|
||||||
@@ -57,9 +57,20 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
included_children = t.profiler[0].children
|
included_children = t.profiler[0].children
|
||||||
|
|
||||||
# {% assign template_name = 'a_template' %}
|
# {% assign template_name = 'a_template' %}
|
||||||
assert_equal 1, included_children[0].line_number
|
assert_equal(1, included_children[0].line_number)
|
||||||
# {{ template_name }}
|
# {{ template_name }}
|
||||||
assert_equal 2, included_children[1].line_number
|
assert_equal(2, included_children[1].line_number)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_profiling_render_tag
|
||||||
|
t = Template.parse("{% render 'a_template' %}", profile: true)
|
||||||
|
t.render!
|
||||||
|
|
||||||
|
render_children = t.profiler[0].children
|
||||||
|
render_children.each do |timing|
|
||||||
|
assert_equal('a_template', timing.partial)
|
||||||
|
end
|
||||||
|
assert_equal([1, 2], render_children.map(&:line_number))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_times_the_rendering_of_tokens
|
def test_profiling_times_the_rendering_of_tokens
|
||||||
@@ -67,14 +78,45 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
t.render!
|
t.render!
|
||||||
|
|
||||||
node = t.profiler[0]
|
node = t.profiler[0]
|
||||||
refute_nil node.render_time
|
refute_nil(node.render_time)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_times_the_entire_render
|
def test_profiling_times_the_entire_render
|
||||||
t = Template.parse("{% include 'a_template' %}", profile: true)
|
t = Template.parse("{% include 'a_template' %}", profile: true)
|
||||||
t.render!
|
t.render!
|
||||||
|
|
||||||
assert t.profiler.total_render_time >= 0, "Total render time was not calculated"
|
assert(t.profiler.total_render_time >= 0, "Total render time was not calculated")
|
||||||
|
end
|
||||||
|
|
||||||
|
class SleepTag < Liquid::Tag
|
||||||
|
def initialize(tag_name, markup, parse_context)
|
||||||
|
super
|
||||||
|
@duration = Float(markup)
|
||||||
|
end
|
||||||
|
|
||||||
|
def render_to_output_buffer(_context, _output)
|
||||||
|
sleep(@duration)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_profiling_multiple_renders
|
||||||
|
with_custom_tag('sleep', SleepTag) do
|
||||||
|
context = Liquid::Context.new
|
||||||
|
t = Liquid::Template.parse("{% sleep 0.001 %}", profile: true)
|
||||||
|
context.template_name = 'index'
|
||||||
|
t.render!(context)
|
||||||
|
context.template_name = 'layout'
|
||||||
|
first_render_time = context.profiler.total_time
|
||||||
|
t.render!(context)
|
||||||
|
|
||||||
|
profiler = context.profiler
|
||||||
|
children = profiler.children
|
||||||
|
assert_operator(first_render_time, :>=, 0.001)
|
||||||
|
assert_operator(profiler.total_time, :>=, 0.001 + first_render_time)
|
||||||
|
assert_equal(["index", "layout"], children.map(&:template_name))
|
||||||
|
assert_equal([nil, nil], children.map(&:code))
|
||||||
|
assert_equal(profiler.total_time, children.map(&:total_time).reduce(&:+))
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_uses_include_to_mark_children
|
def test_profiling_uses_include_to_mark_children
|
||||||
@@ -82,7 +124,7 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
t.render!
|
t.render!
|
||||||
|
|
||||||
include_node = t.profiler[1]
|
include_node = t.profiler[1]
|
||||||
assert_equal 2, include_node.children.length
|
assert_equal(2, include_node.children.length)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_marks_children_with_the_name_of_included_partial
|
def test_profiling_marks_children_with_the_name_of_included_partial
|
||||||
@@ -91,7 +133,7 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
|
|
||||||
include_node = t.profiler[1]
|
include_node = t.profiler[1]
|
||||||
include_node.children.each do |child|
|
include_node.children.each do |child|
|
||||||
assert_equal "a_template", child.partial
|
assert_equal("a_template", child.partial)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -101,12 +143,12 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
|
|
||||||
a_template = t.profiler[1]
|
a_template = t.profiler[1]
|
||||||
a_template.children.each do |child|
|
a_template.children.each do |child|
|
||||||
assert_equal "a_template", child.partial
|
assert_equal("a_template", child.partial)
|
||||||
end
|
end
|
||||||
|
|
||||||
b_template = t.profiler[2]
|
b_template = t.profiler[2]
|
||||||
b_template.children.each do |child|
|
b_template.children.each do |child|
|
||||||
assert_equal "b_template", child.partial
|
assert_equal("b_template", child.partial)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -116,12 +158,12 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
|
|
||||||
a_template1 = t.profiler[1]
|
a_template1 = t.profiler[1]
|
||||||
a_template1.children.each do |child|
|
a_template1.children.each do |child|
|
||||||
assert_equal "a_template", child.partial
|
assert_equal("a_template", child.partial)
|
||||||
end
|
end
|
||||||
|
|
||||||
a_template2 = t.profiler[2]
|
a_template2 = t.profiler[2]
|
||||||
a_template2.children.each do |child|
|
a_template2.children.each do |child|
|
||||||
assert_equal "a_template", child.partial
|
assert_equal("a_template", child.partial)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -134,23 +176,38 @@ class RenderProfilingTest < Minitest::Test
|
|||||||
timing_count += 1
|
timing_count += 1
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_equal 2, timing_count
|
assert_equal(2, timing_count)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_marks_children_of_if_blocks
|
def test_profiling_marks_children_of_if_blocks
|
||||||
t = Template.parse("{% if true %} {% increment test %} {{ test }} {% endif %}", profile: true)
|
t = Template.parse("{% if true %} {% increment test %} {{ test }} {% endif %}", profile: true)
|
||||||
t.render!
|
t.render!
|
||||||
|
|
||||||
assert_equal 1, t.profiler.length
|
assert_equal(1, t.profiler.length)
|
||||||
assert_equal 2, t.profiler[0].children.length
|
assert_equal(2, t.profiler[0].children.length)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_profiling_marks_children_of_for_blocks
|
def test_profiling_marks_children_of_for_blocks
|
||||||
t = Template.parse("{% for item in collection %} {{ item }} {% endfor %}", profile: true)
|
t = Template.parse("{% for item in collection %} {{ item }} {% endfor %}", profile: true)
|
||||||
t.render!("collection" => ["one", "two"])
|
t.render!("collection" => ["one", "two"])
|
||||||
|
|
||||||
assert_equal 1, t.profiler.length
|
assert_equal(1, t.profiler.length)
|
||||||
# Will profile each invocation of the for block
|
# Will profile each invocation of the for block
|
||||||
assert_equal 2, t.profiler[0].children.length
|
assert_equal(2, t.profiler[0].children.length)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_profiling_supports_self_time
|
||||||
|
t = Template.parse("{% for item in collection %} {{ item }} {% endfor %}", profile: true)
|
||||||
|
t.render!("collection" => ["one", "two"])
|
||||||
|
leaf = t.profiler[0].children[0]
|
||||||
|
|
||||||
|
assert_operator(leaf.self_time, :>, 0)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_profiling_supports_total_time
|
||||||
|
t = Template.parse("{% if true %} {% increment test %} {{ test }} {% endif %}", profile: true)
|
||||||
|
t.render!
|
||||||
|
|
||||||
|
assert_operator(t.profiler[0].total_time, :>, 0)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
# frozen_string_literal: true
|
|
||||||
|
|
||||||
require 'test_helper'
|
|
||||||
|
|
||||||
class DisabledTagsTest < Minitest::Test
|
|
||||||
include Liquid
|
|
||||||
|
|
||||||
class DisableRaw < Block
|
|
||||||
disable_tags "raw"
|
|
||||||
end
|
|
||||||
|
|
||||||
class DisableRawEcho < Block
|
|
||||||
disable_tags "raw", "echo"
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_disables_raw
|
|
||||||
with_custom_tag('disable', DisableRaw) do
|
|
||||||
assert_template_result 'raw usage is not allowed in this contextfoo', '{% disable %}{% raw %}Foobar{% endraw %}{% echo "foo" %}{% enddisable %}'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_disables_echo_and_raw
|
|
||||||
with_custom_tag('disable', DisableRawEcho) do
|
|
||||||
assert_template_result 'raw usage is not allowed in this contextecho usage is not allowed in this context', '{% disable %}{% raw %}Foobar{% endraw %}{% echo "foo" %}{% enddisable %}'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -19,57 +19,64 @@ class SecurityTest < Minitest::Test
|
|||||||
text = %( {{ '1+1' | instance_eval }} )
|
text = %( {{ '1+1' | instance_eval }} )
|
||||||
expected = %( 1+1 )
|
expected = %( 1+1 )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns)
|
assert_equal(expected, Template.parse(text).render!(@assigns))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_no_existing_instance_eval
|
def test_no_existing_instance_eval
|
||||||
text = %( {{ '1+1' | __instance_eval__ }} )
|
text = %( {{ '1+1' | __instance_eval__ }} )
|
||||||
expected = %( 1+1 )
|
expected = %( 1+1 )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns)
|
assert_equal(expected, Template.parse(text).render!(@assigns))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_no_instance_eval_after_mixing_in_new_filter
|
def test_no_instance_eval_after_mixing_in_new_filter
|
||||||
text = %( {{ '1+1' | instance_eval }} )
|
text = %( {{ '1+1' | instance_eval }} )
|
||||||
expected = %( 1+1 )
|
expected = %( 1+1 )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns)
|
assert_equal(expected, Template.parse(text).render!(@assigns))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_no_instance_eval_later_in_chain
|
def test_no_instance_eval_later_in_chain
|
||||||
text = %( {{ '1+1' | add_one | instance_eval }} )
|
text = %( {{ '1+1' | add_one | instance_eval }} )
|
||||||
expected = %( 1+1 + 1 )
|
expected = %( 1+1 + 1 )
|
||||||
|
|
||||||
assert_equal expected, Template.parse(text).render!(@assigns, filters: SecurityFilter)
|
assert_equal(expected, Template.parse(text).render!(@assigns, filters: SecurityFilter))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_does_not_add_filters_to_symbol_table
|
def test_does_not_permanently_add_filters_to_symbol_table
|
||||||
current_symbols = Symbol.all_symbols
|
current_symbols = Symbol.all_symbols
|
||||||
|
|
||||||
|
# MRI imprecisely marks objects found on the C stack, which can result
|
||||||
|
# in uninitialized memory being marked. This can even result in the test failing
|
||||||
|
# deterministically for a given compilation of ruby. Using a separate thread will
|
||||||
|
# keep these writes of the symbol pointer on a separate stack that will be garbage
|
||||||
|
# collected after Thread#join.
|
||||||
|
Thread.new do
|
||||||
test = %( {{ "some_string" | a_bad_filter }} )
|
test = %( {{ "some_string" | a_bad_filter }} )
|
||||||
|
Template.parse(test).render!
|
||||||
|
nil
|
||||||
|
end.join
|
||||||
|
|
||||||
template = Template.parse(test)
|
GC.start
|
||||||
assert_equal [], (Symbol.all_symbols - current_symbols)
|
|
||||||
|
|
||||||
template.render!
|
assert_equal([], (Symbol.all_symbols - current_symbols))
|
||||||
assert_equal [], (Symbol.all_symbols - current_symbols)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_does_not_add_drop_methods_to_symbol_table
|
def test_does_not_add_drop_methods_to_symbol_table
|
||||||
current_symbols = Symbol.all_symbols
|
current_symbols = Symbol.all_symbols
|
||||||
|
|
||||||
assigns = { 'drop' => Drop.new }
|
assigns = { 'drop' => Drop.new }
|
||||||
assert_equal "", Template.parse("{{ drop.custom_method_1 }}", assigns).render!
|
assert_equal("", Template.parse("{{ drop.custom_method_1 }}", assigns).render!)
|
||||||
assert_equal "", Template.parse("{{ drop.custom_method_2 }}", assigns).render!
|
assert_equal("", Template.parse("{{ drop.custom_method_2 }}", assigns).render!)
|
||||||
assert_equal "", Template.parse("{{ drop.custom_method_3 }}", assigns).render!
|
assert_equal("", Template.parse("{{ drop.custom_method_3 }}", assigns).render!)
|
||||||
|
|
||||||
assert_equal [], (Symbol.all_symbols - current_symbols)
|
assert_equal([], (Symbol.all_symbols - current_symbols))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_max_depth_nested_blocks_does_not_raise_exception
|
def test_max_depth_nested_blocks_does_not_raise_exception
|
||||||
depth = Liquid::Block::MAX_DEPTH
|
depth = Liquid::Block::MAX_DEPTH
|
||||||
code = "{% if true %}" * depth + "rendered" + "{% endif %}" * depth
|
code = "{% if true %}" * depth + "rendered" + "{% endif %}" * depth
|
||||||
assert_equal "rendered", Template.parse(code).render!
|
assert_equal("rendered", Template.parse(code).render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_more_than_max_depth_nested_blocks_raises_exception
|
def test_more_than_max_depth_nested_blocks_raises_exception
|
||||||
|
|||||||
@@ -60,34 +60,34 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_size
|
def test_size
|
||||||
assert_equal 3, @filters.size([1, 2, 3])
|
assert_equal(3, @filters.size([1, 2, 3]))
|
||||||
assert_equal 0, @filters.size([])
|
assert_equal(0, @filters.size([]))
|
||||||
assert_equal 0, @filters.size(nil)
|
assert_equal(0, @filters.size(nil))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_downcase
|
def test_downcase
|
||||||
assert_equal 'testing', @filters.downcase("Testing")
|
assert_equal('testing', @filters.downcase("Testing"))
|
||||||
assert_equal '', @filters.downcase(nil)
|
assert_equal('', @filters.downcase(nil))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_upcase
|
def test_upcase
|
||||||
assert_equal 'TESTING', @filters.upcase("Testing")
|
assert_equal('TESTING', @filters.upcase("Testing"))
|
||||||
assert_equal '', @filters.upcase(nil)
|
assert_equal('', @filters.upcase(nil))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_slice
|
def test_slice
|
||||||
assert_equal 'oob', @filters.slice('foobar', 1, 3)
|
assert_equal('oob', @filters.slice('foobar', 1, 3))
|
||||||
assert_equal 'oobar', @filters.slice('foobar', 1, 1000)
|
assert_equal('oobar', @filters.slice('foobar', 1, 1000))
|
||||||
assert_equal '', @filters.slice('foobar', 1, 0)
|
assert_equal('', @filters.slice('foobar', 1, 0))
|
||||||
assert_equal 'o', @filters.slice('foobar', 1, 1)
|
assert_equal('o', @filters.slice('foobar', 1, 1))
|
||||||
assert_equal 'bar', @filters.slice('foobar', 3, 3)
|
assert_equal('bar', @filters.slice('foobar', 3, 3))
|
||||||
assert_equal 'ar', @filters.slice('foobar', -2, 2)
|
assert_equal('ar', @filters.slice('foobar', -2, 2))
|
||||||
assert_equal 'ar', @filters.slice('foobar', -2, 1000)
|
assert_equal('ar', @filters.slice('foobar', -2, 1000))
|
||||||
assert_equal 'r', @filters.slice('foobar', -1)
|
assert_equal('r', @filters.slice('foobar', -1))
|
||||||
assert_equal '', @filters.slice(nil, 0)
|
assert_equal('', @filters.slice(nil, 0))
|
||||||
assert_equal '', @filters.slice('foobar', 100, 10)
|
assert_equal('', @filters.slice('foobar', 100, 10))
|
||||||
assert_equal '', @filters.slice('foobar', -100, 10)
|
assert_equal('', @filters.slice('foobar', -100, 10))
|
||||||
assert_equal 'oob', @filters.slice('foobar', '1', '3')
|
assert_equal('oob', @filters.slice('foobar', '1', '3'))
|
||||||
assert_raises(Liquid::ArgumentError) do
|
assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.slice('foobar', nil)
|
@filters.slice('foobar', nil)
|
||||||
end
|
end
|
||||||
@@ -98,109 +98,115 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
|
|
||||||
def test_slice_on_arrays
|
def test_slice_on_arrays
|
||||||
input = 'foobar'.split(//)
|
input = 'foobar'.split(//)
|
||||||
assert_equal %w(o o b), @filters.slice(input, 1, 3)
|
assert_equal(%w(o o b), @filters.slice(input, 1, 3))
|
||||||
assert_equal %w(o o b a r), @filters.slice(input, 1, 1000)
|
assert_equal(%w(o o b a r), @filters.slice(input, 1, 1000))
|
||||||
assert_equal %w(), @filters.slice(input, 1, 0)
|
assert_equal(%w(), @filters.slice(input, 1, 0))
|
||||||
assert_equal %w(o), @filters.slice(input, 1, 1)
|
assert_equal(%w(o), @filters.slice(input, 1, 1))
|
||||||
assert_equal %w(b a r), @filters.slice(input, 3, 3)
|
assert_equal(%w(b a r), @filters.slice(input, 3, 3))
|
||||||
assert_equal %w(a r), @filters.slice(input, -2, 2)
|
assert_equal(%w(a r), @filters.slice(input, -2, 2))
|
||||||
assert_equal %w(a r), @filters.slice(input, -2, 1000)
|
assert_equal(%w(a r), @filters.slice(input, -2, 1000))
|
||||||
assert_equal %w(r), @filters.slice(input, -1)
|
assert_equal(%w(r), @filters.slice(input, -1))
|
||||||
assert_equal %w(), @filters.slice(input, 100, 10)
|
assert_equal(%w(), @filters.slice(input, 100, 10))
|
||||||
assert_equal %w(), @filters.slice(input, -100, 10)
|
assert_equal(%w(), @filters.slice(input, -100, 10))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_truncate
|
def test_truncate
|
||||||
assert_equal '1234...', @filters.truncate('1234567890', 7)
|
assert_equal('1234...', @filters.truncate('1234567890', 7))
|
||||||
assert_equal '1234567890', @filters.truncate('1234567890', 20)
|
assert_equal('1234567890', @filters.truncate('1234567890', 20))
|
||||||
assert_equal '...', @filters.truncate('1234567890', 0)
|
assert_equal('...', @filters.truncate('1234567890', 0))
|
||||||
assert_equal '1234567890', @filters.truncate('1234567890')
|
assert_equal('1234567890', @filters.truncate('1234567890'))
|
||||||
assert_equal "测试...", @filters.truncate("测试测试测试测试", 5)
|
assert_equal("测试...", @filters.truncate("测试测试测试测试", 5))
|
||||||
assert_equal '12341', @filters.truncate("1234567890", 5, 1)
|
assert_equal('12341', @filters.truncate("1234567890", 5, 1))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_split
|
def test_split
|
||||||
assert_equal ['12', '34'], @filters.split('12~34', '~')
|
assert_equal(['12', '34'], @filters.split('12~34', '~'))
|
||||||
assert_equal ['A? ', ' ,Z'], @filters.split('A? ~ ~ ~ ,Z', '~ ~ ~')
|
assert_equal(['A? ', ' ,Z'], @filters.split('A? ~ ~ ~ ,Z', '~ ~ ~'))
|
||||||
assert_equal ['A?Z'], @filters.split('A?Z', '~')
|
assert_equal(['A?Z'], @filters.split('A?Z', '~'))
|
||||||
assert_equal [], @filters.split(nil, ' ')
|
assert_equal([], @filters.split(nil, ' '))
|
||||||
assert_equal ['A', 'Z'], @filters.split('A1Z', 1)
|
assert_equal(['A', 'Z'], @filters.split('A1Z', 1))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_escape
|
def test_escape
|
||||||
assert_equal '<strong>', @filters.escape('<strong>')
|
assert_equal('<strong>', @filters.escape('<strong>'))
|
||||||
assert_equal '1', @filters.escape(1)
|
assert_equal('1', @filters.escape(1))
|
||||||
assert_equal '2001-02-03', @filters.escape(Date.new(2001, 2, 3))
|
assert_equal('2001-02-03', @filters.escape(Date.new(2001, 2, 3)))
|
||||||
assert_nil @filters.escape(nil)
|
assert_nil(@filters.escape(nil))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_h
|
def test_h
|
||||||
assert_equal '<strong>', @filters.h('<strong>')
|
assert_equal('<strong>', @filters.h('<strong>'))
|
||||||
assert_equal '1', @filters.h(1)
|
assert_equal('1', @filters.h(1))
|
||||||
assert_equal '2001-02-03', @filters.h(Date.new(2001, 2, 3))
|
assert_equal('2001-02-03', @filters.h(Date.new(2001, 2, 3)))
|
||||||
assert_nil @filters.h(nil)
|
assert_nil(@filters.h(nil))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_escape_once
|
def test_escape_once
|
||||||
assert_equal '<strong>Hulk</strong>', @filters.escape_once('<strong>Hulk</strong>')
|
assert_equal('<strong>Hulk</strong>', @filters.escape_once('<strong>Hulk</strong>'))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_url_encode
|
def test_url_encode
|
||||||
assert_equal 'foo%2B1%40example.com', @filters.url_encode('foo+1@example.com')
|
assert_equal('foo%2B1%40example.com', @filters.url_encode('foo+1@example.com'))
|
||||||
assert_equal '1', @filters.url_encode(1)
|
assert_equal('1', @filters.url_encode(1))
|
||||||
assert_equal '2001-02-03', @filters.url_encode(Date.new(2001, 2, 3))
|
assert_equal('2001-02-03', @filters.url_encode(Date.new(2001, 2, 3)))
|
||||||
assert_nil @filters.url_encode(nil)
|
assert_nil(@filters.url_encode(nil))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_url_decode
|
def test_url_decode
|
||||||
assert_equal 'foo bar', @filters.url_decode('foo+bar')
|
assert_equal('foo bar', @filters.url_decode('foo+bar'))
|
||||||
assert_equal 'foo bar', @filters.url_decode('foo%20bar')
|
assert_equal('foo bar', @filters.url_decode('foo%20bar'))
|
||||||
assert_equal 'foo+1@example.com', @filters.url_decode('foo%2B1%40example.com')
|
assert_equal('foo+1@example.com', @filters.url_decode('foo%2B1%40example.com'))
|
||||||
assert_equal '1', @filters.url_decode(1)
|
assert_equal('1', @filters.url_decode(1))
|
||||||
assert_equal '2001-02-03', @filters.url_decode(Date.new(2001, 2, 3))
|
assert_equal('2001-02-03', @filters.url_decode(Date.new(2001, 2, 3)))
|
||||||
assert_nil @filters.url_decode(nil)
|
assert_nil(@filters.url_decode(nil))
|
||||||
exception = assert_raises Liquid::ArgumentError do
|
exception = assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.url_decode('%ff')
|
@filters.url_decode('%ff')
|
||||||
end
|
end
|
||||||
assert_equal 'Liquid error: invalid byte sequence in UTF-8', exception.message
|
assert_equal('Liquid error: invalid byte sequence in UTF-8', exception.message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_truncatewords
|
def test_truncatewords
|
||||||
assert_equal 'one two three', @filters.truncatewords('one two three', 4)
|
assert_equal('one two three', @filters.truncatewords('one two three', 4))
|
||||||
assert_equal 'one two...', @filters.truncatewords('one two three', 2)
|
assert_equal('one two...', @filters.truncatewords('one two three', 2))
|
||||||
assert_equal 'one two three', @filters.truncatewords('one two three')
|
assert_equal('one two three', @filters.truncatewords('one two three'))
|
||||||
assert_equal 'Two small (13” x 5.5” x 10” high) baskets fit inside one large basket (13”...', @filters.truncatewords('Two small (13” x 5.5” x 10” high) baskets fit inside one large basket (13” x 16” x 10.5” high) with cover.', 15)
|
assert_equal(
|
||||||
assert_equal "测试测试测试测试", @filters.truncatewords('测试测试测试测试', 5)
|
'Two small (13” x 5.5” x 10” high) baskets fit inside one large basket (13”...',
|
||||||
assert_equal 'one two1', @filters.truncatewords("one two three", 2, 1)
|
@filters.truncatewords('Two small (13” x 5.5” x 10” high) baskets fit inside one large basket (13” x 16” x 10.5” high) with cover.', 15)
|
||||||
|
)
|
||||||
|
assert_equal("测试测试测试测试", @filters.truncatewords('测试测试测试测试', 5))
|
||||||
|
assert_equal('one two1', @filters.truncatewords("one two three", 2, 1))
|
||||||
|
assert_equal('one two three...', @filters.truncatewords("one two\tthree\nfour", 3))
|
||||||
|
assert_equal('one two...', @filters.truncatewords("one two three four", 2))
|
||||||
|
assert_equal('one...', @filters.truncatewords("one two three four", 0))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strip_html
|
def test_strip_html
|
||||||
assert_equal 'test', @filters.strip_html("<div>test</div>")
|
assert_equal('test', @filters.strip_html("<div>test</div>"))
|
||||||
assert_equal 'test', @filters.strip_html("<div id='test'>test</div>")
|
assert_equal('test', @filters.strip_html("<div id='test'>test</div>"))
|
||||||
assert_equal '', @filters.strip_html("<script type='text/javascript'>document.write('some stuff');</script>")
|
assert_equal('', @filters.strip_html("<script type='text/javascript'>document.write('some stuff');</script>"))
|
||||||
assert_equal '', @filters.strip_html("<style type='text/css'>foo bar</style>")
|
assert_equal('', @filters.strip_html("<style type='text/css'>foo bar</style>"))
|
||||||
assert_equal 'test', @filters.strip_html("<div\nclass='multiline'>test</div>")
|
assert_equal('test', @filters.strip_html("<div\nclass='multiline'>test</div>"))
|
||||||
assert_equal 'test', @filters.strip_html("<!-- foo bar \n test -->test")
|
assert_equal('test', @filters.strip_html("<!-- foo bar \n test -->test"))
|
||||||
assert_equal '', @filters.strip_html(nil)
|
assert_equal('', @filters.strip_html(nil))
|
||||||
|
|
||||||
# Quirk of the existing implementation
|
# Quirk of the existing implementation
|
||||||
assert_equal 'foo;', @filters.strip_html("<<<script </script>script>foo;</script>")
|
assert_equal('foo;', @filters.strip_html("<<<script </script>script>foo;</script>"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_join
|
def test_join
|
||||||
assert_equal '1 2 3 4', @filters.join([1, 2, 3, 4])
|
assert_equal('1 2 3 4', @filters.join([1, 2, 3, 4]))
|
||||||
assert_equal '1 - 2 - 3 - 4', @filters.join([1, 2, 3, 4], ' - ')
|
assert_equal('1 - 2 - 3 - 4', @filters.join([1, 2, 3, 4], ' - '))
|
||||||
assert_equal '1121314', @filters.join([1, 2, 3, 4], 1)
|
assert_equal('1121314', @filters.join([1, 2, 3, 4], 1))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort
|
def test_sort
|
||||||
assert_equal [1, 2, 3, 4], @filters.sort([4, 3, 2, 1])
|
assert_equal([1, 2, 3, 4], @filters.sort([4, 3, 2, 1]))
|
||||||
assert_equal [{ "a" => 1 }, { "a" => 2 }, { "a" => 3 }, { "a" => 4 }], @filters.sort([{ "a" => 4 }, { "a" => 3 }, { "a" => 1 }, { "a" => 2 }], "a")
|
assert_equal([{ "a" => 1 }, { "a" => 2 }, { "a" => 3 }, { "a" => 4 }], @filters.sort([{ "a" => 4 }, { "a" => 3 }, { "a" => 1 }, { "a" => 2 }], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_with_nils
|
def test_sort_with_nils
|
||||||
assert_equal [1, 2, 3, 4, nil], @filters.sort([nil, 4, 3, 2, 1])
|
assert_equal([1, 2, 3, 4, nil], @filters.sort([nil, 4, 3, 2, 1]))
|
||||||
assert_equal [{ "a" => 1 }, { "a" => 2 }, { "a" => 3 }, { "a" => 4 }, {}], @filters.sort([{ "a" => 4 }, { "a" => 3 }, {}, { "a" => 1 }, { "a" => 2 }], "a")
|
assert_equal([{ "a" => 1 }, { "a" => 2 }, { "a" => 3 }, { "a" => 4 }, {}], @filters.sort([{ "a" => 4 }, { "a" => 3 }, {}, { "a" => 1 }, { "a" => 2 }], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_when_property_is_sometimes_missing_puts_nils_last
|
def test_sort_when_property_is_sometimes_missing_puts_nils_last
|
||||||
@@ -218,17 +224,17 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
{ "handle" => "delta" },
|
{ "handle" => "delta" },
|
||||||
{ "handle" => "beta" },
|
{ "handle" => "beta" },
|
||||||
]
|
]
|
||||||
assert_equal expectation, @filters.sort(input, "price")
|
assert_equal(expectation, @filters.sort(input, "price"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_natural
|
def test_sort_natural
|
||||||
assert_equal ["a", "B", "c", "D"], @filters.sort_natural(["c", "D", "a", "B"])
|
assert_equal(["a", "B", "c", "D"], @filters.sort_natural(["c", "D", "a", "B"]))
|
||||||
assert_equal [{ "a" => "a" }, { "a" => "B" }, { "a" => "c" }, { "a" => "D" }], @filters.sort_natural([{ "a" => "D" }, { "a" => "c" }, { "a" => "a" }, { "a" => "B" }], "a")
|
assert_equal([{ "a" => "a" }, { "a" => "B" }, { "a" => "c" }, { "a" => "D" }], @filters.sort_natural([{ "a" => "D" }, { "a" => "c" }, { "a" => "a" }, { "a" => "B" }], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_natural_with_nils
|
def test_sort_natural_with_nils
|
||||||
assert_equal ["a", "B", "c", "D", nil], @filters.sort_natural([nil, "c", "D", "a", "B"])
|
assert_equal(["a", "B", "c", "D", nil], @filters.sort_natural([nil, "c", "D", "a", "B"]))
|
||||||
assert_equal [{ "a" => "a" }, { "a" => "B" }, { "a" => "c" }, { "a" => "D" }, {}], @filters.sort_natural([{ "a" => "D" }, { "a" => "c" }, {}, { "a" => "a" }, { "a" => "B" }], "a")
|
assert_equal([{ "a" => "a" }, { "a" => "B" }, { "a" => "c" }, { "a" => "D" }, {}], @filters.sort_natural([{ "a" => "D" }, { "a" => "c" }, {}, { "a" => "a" }, { "a" => "B" }], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_natural_when_property_is_sometimes_missing_puts_nils_last
|
def test_sort_natural_when_property_is_sometimes_missing_puts_nils_last
|
||||||
@@ -246,7 +252,7 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
{ "handle" => "delta" },
|
{ "handle" => "delta" },
|
||||||
{ "handle" => "beta" },
|
{ "handle" => "beta" },
|
||||||
]
|
]
|
||||||
assert_equal expectation, @filters.sort_natural(input, "price")
|
assert_equal(expectation, @filters.sort_natural(input, "price"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_natural_case_check
|
def test_sort_natural_case_check
|
||||||
@@ -268,12 +274,12 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
{ "key" => "Z" },
|
{ "key" => "Z" },
|
||||||
{ "fake" => "t" },
|
{ "fake" => "t" },
|
||||||
]
|
]
|
||||||
assert_equal expectation, @filters.sort_natural(input, "key")
|
assert_equal(expectation, @filters.sort_natural(input, "key"))
|
||||||
assert_equal ["a", "b", "c", "X", "Y", "Z"], @filters.sort_natural(["X", "Y", "Z", "a", "b", "c"])
|
assert_equal(["a", "b", "c", "X", "Y", "Z"], @filters.sort_natural(["X", "Y", "Z", "a", "b", "c"]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_empty_array
|
def test_sort_empty_array
|
||||||
assert_equal [], @filters.sort([], "a")
|
assert_equal([], @filters.sort([], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_invalid_property
|
def test_sort_invalid_property
|
||||||
@@ -283,13 +289,13 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
[3],
|
[3],
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_raises Liquid::ArgumentError do
|
assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.sort(foo, "bar")
|
@filters.sort(foo, "bar")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_natural_empty_array
|
def test_sort_natural_empty_array
|
||||||
assert_equal [], @filters.sort_natural([], "a")
|
assert_equal([], @filters.sort_natural([], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_natural_invalid_property
|
def test_sort_natural_invalid_property
|
||||||
@@ -299,32 +305,32 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
[3],
|
[3],
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_raises Liquid::ArgumentError do
|
assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.sort_natural(foo, "bar")
|
@filters.sort_natural(foo, "bar")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_legacy_sort_hash
|
def test_legacy_sort_hash
|
||||||
assert_equal [{ a: 1, b: 2 }], @filters.sort(a: 1, b: 2)
|
assert_equal([{ a: 1, b: 2 }], @filters.sort(a: 1, b: 2))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_numerical_vs_lexicographical_sort
|
def test_numerical_vs_lexicographical_sort
|
||||||
assert_equal [2, 10], @filters.sort([10, 2])
|
assert_equal([2, 10], @filters.sort([10, 2]))
|
||||||
assert_equal [{ "a" => 2 }, { "a" => 10 }], @filters.sort([{ "a" => 10 }, { "a" => 2 }], "a")
|
assert_equal([{ "a" => 2 }, { "a" => 10 }], @filters.sort([{ "a" => 10 }, { "a" => 2 }], "a"))
|
||||||
assert_equal ["10", "2"], @filters.sort(["10", "2"])
|
assert_equal(["10", "2"], @filters.sort(["10", "2"]))
|
||||||
assert_equal [{ "a" => "10" }, { "a" => "2" }], @filters.sort([{ "a" => "10" }, { "a" => "2" }], "a")
|
assert_equal([{ "a" => "10" }, { "a" => "2" }], @filters.sort([{ "a" => "10" }, { "a" => "2" }], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_uniq
|
def test_uniq
|
||||||
assert_equal ["foo"], @filters.uniq("foo")
|
assert_equal(["foo"], @filters.uniq("foo"))
|
||||||
assert_equal [1, 3, 2, 4], @filters.uniq([1, 1, 3, 2, 3, 1, 4, 3, 2, 1])
|
assert_equal([1, 3, 2, 4], @filters.uniq([1, 1, 3, 2, 3, 1, 4, 3, 2, 1]))
|
||||||
assert_equal [{ "a" => 1 }, { "a" => 3 }, { "a" => 2 }], @filters.uniq([{ "a" => 1 }, { "a" => 3 }, { "a" => 1 }, { "a" => 2 }], "a")
|
assert_equal([{ "a" => 1 }, { "a" => 3 }, { "a" => 2 }], @filters.uniq([{ "a" => 1 }, { "a" => 3 }, { "a" => 1 }, { "a" => 2 }], "a"))
|
||||||
testdrop = TestDrop.new
|
testdrop = TestDrop.new
|
||||||
assert_equal [testdrop], @filters.uniq([testdrop, TestDrop.new], 'test')
|
assert_equal([testdrop], @filters.uniq([testdrop, TestDrop.new], 'test'))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_uniq_empty_array
|
def test_uniq_empty_array
|
||||||
assert_equal [], @filters.uniq([], "a")
|
assert_equal([], @filters.uniq([], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_uniq_invalid_property
|
def test_uniq_invalid_property
|
||||||
@@ -334,13 +340,13 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
[3],
|
[3],
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_raises Liquid::ArgumentError do
|
assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.uniq(foo, "bar")
|
@filters.uniq(foo, "bar")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_compact_empty_array
|
def test_compact_empty_array
|
||||||
assert_equal [], @filters.compact([], "a")
|
assert_equal([], @filters.compact([], "a"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_compact_invalid_property
|
def test_compact_invalid_property
|
||||||
@@ -350,57 +356,57 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
[3],
|
[3],
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_raises Liquid::ArgumentError do
|
assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.compact(foo, "bar")
|
@filters.compact(foo, "bar")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_reverse
|
def test_reverse
|
||||||
assert_equal [4, 3, 2, 1], @filters.reverse([1, 2, 3, 4])
|
assert_equal([4, 3, 2, 1], @filters.reverse([1, 2, 3, 4]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_legacy_reverse_hash
|
def test_legacy_reverse_hash
|
||||||
assert_equal [{ a: 1, b: 2 }], @filters.reverse(a: 1, b: 2)
|
assert_equal([{ a: 1, b: 2 }], @filters.reverse(a: 1, b: 2))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map
|
def test_map
|
||||||
assert_equal [1, 2, 3, 4], @filters.map([{ "a" => 1 }, { "a" => 2 }, { "a" => 3 }, { "a" => 4 }], 'a')
|
assert_equal([1, 2, 3, 4], @filters.map([{ "a" => 1 }, { "a" => 2 }, { "a" => 3 }, { "a" => 4 }], 'a'))
|
||||||
assert_template_result 'abc', "{{ ary | map:'foo' | map:'bar' }}",
|
assert_template_result('abc', "{{ ary | map:'foo' | map:'bar' }}",
|
||||||
'ary' => [{ 'foo' => { 'bar' => 'a' } }, { 'foo' => { 'bar' => 'b' } }, { 'foo' => { 'bar' => 'c' } }]
|
'ary' => [{ 'foo' => { 'bar' => 'a' } }, { 'foo' => { 'bar' => 'b' } }, { 'foo' => { 'bar' => 'c' } }])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map_doesnt_call_arbitrary_stuff
|
def test_map_doesnt_call_arbitrary_stuff
|
||||||
assert_template_result "", '{{ "foo" | map: "__id__" }}'
|
assert_template_result("", '{{ "foo" | map: "__id__" }}')
|
||||||
assert_template_result "", '{{ "foo" | map: "inspect" }}'
|
assert_template_result("", '{{ "foo" | map: "inspect" }}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map_calls_to_liquid
|
def test_map_calls_to_liquid
|
||||||
t = TestThing.new
|
t = TestThing.new
|
||||||
assert_template_result "woot: 1", '{{ foo | map: "whatever" }}', "foo" => [t]
|
assert_template_result("woot: 1", '{{ foo | map: "whatever" }}', "foo" => [t])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map_on_hashes
|
def test_map_on_hashes
|
||||||
assert_template_result "4217", '{{ thing | map: "foo" | map: "bar" }}',
|
assert_template_result("4217", '{{ thing | map: "foo" | map: "bar" }}',
|
||||||
"thing" => { "foo" => [{ "bar" => 42 }, { "bar" => 17 }] }
|
"thing" => { "foo" => [{ "bar" => 42 }, { "bar" => 17 }] })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_legacy_map_on_hashes_with_dynamic_key
|
def test_legacy_map_on_hashes_with_dynamic_key
|
||||||
template = "{% assign key = 'foo' %}{{ thing | map: key | map: 'bar' }}"
|
template = "{% assign key = 'foo' %}{{ thing | map: key | map: 'bar' }}"
|
||||||
hash = { "foo" => { "bar" => 42 } }
|
hash = { "foo" => { "bar" => 42 } }
|
||||||
assert_template_result "42", template, "thing" => hash
|
assert_template_result("42", template, "thing" => hash)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_calls_to_liquid
|
def test_sort_calls_to_liquid
|
||||||
t = TestThing.new
|
t = TestThing.new
|
||||||
Liquid::Template.parse('{{ foo | sort: "whatever" }}').render("foo" => [t])
|
Liquid::Template.parse('{{ foo | sort: "whatever" }}').render("foo" => [t])
|
||||||
assert t.foo > 0
|
assert(t.foo > 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map_over_proc
|
def test_map_over_proc
|
||||||
drop = TestDrop.new
|
drop = TestDrop.new
|
||||||
p = proc { drop }
|
p = proc { drop }
|
||||||
templ = '{{ procs | map: "test" }}'
|
templ = '{{ procs | map: "test" }}'
|
||||||
assert_template_result "testfoo", templ, "procs" => [p]
|
assert_template_result("testfoo", templ, "procs" => [p])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map_over_drops_returning_procs
|
def test_map_over_drops_returning_procs
|
||||||
@@ -413,11 +419,11 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
},
|
},
|
||||||
]
|
]
|
||||||
templ = '{{ drops | map: "proc" }}'
|
templ = '{{ drops | map: "proc" }}'
|
||||||
assert_template_result "foobar", templ, "drops" => drops
|
assert_template_result("foobar", templ, "drops" => drops)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map_works_on_enumerables
|
def test_map_works_on_enumerables
|
||||||
assert_template_result "123", '{{ foo | map: "foo" }}', "foo" => TestEnumerable.new
|
assert_template_result("123", '{{ foo | map: "foo" }}', "foo" => TestEnumerable.new)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_map_returns_empty_on_2d_input_array
|
def test_map_returns_empty_on_2d_input_array
|
||||||
@@ -427,7 +433,7 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
[3],
|
[3],
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_raises Liquid::ArgumentError do
|
assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.map(foo, "bar")
|
@filters.map(foo, "bar")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -438,219 +444,220 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
[2],
|
[2],
|
||||||
[3],
|
[3],
|
||||||
]
|
]
|
||||||
assert_raises Liquid::ArgumentError do
|
assert_raises(Liquid::ArgumentError) do
|
||||||
@filters.map(foo, nil)
|
@filters.map(foo, nil)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sort_works_on_enumerables
|
def test_sort_works_on_enumerables
|
||||||
assert_template_result "213", '{{ foo | sort: "bar" | map: "foo" }}', "foo" => TestEnumerable.new
|
assert_template_result("213", '{{ foo | sort: "bar" | map: "foo" }}', "foo" => TestEnumerable.new)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_first_and_last_call_to_liquid
|
def test_first_and_last_call_to_liquid
|
||||||
assert_template_result 'foobar', '{{ foo | first }}', 'foo' => [ThingWithToLiquid.new]
|
assert_template_result('foobar', '{{ foo | first }}', 'foo' => [ThingWithToLiquid.new])
|
||||||
assert_template_result 'foobar', '{{ foo | last }}', 'foo' => [ThingWithToLiquid.new]
|
assert_template_result('foobar', '{{ foo | last }}', 'foo' => [ThingWithToLiquid.new])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_truncate_calls_to_liquid
|
def test_truncate_calls_to_liquid
|
||||||
assert_template_result "wo...", '{{ foo | truncate: 5 }}', "foo" => TestThing.new
|
assert_template_result("wo...", '{{ foo | truncate: 5 }}', "foo" => TestThing.new)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_date
|
def test_date
|
||||||
assert_equal 'May', @filters.date(Time.parse("2006-05-05 10:00:00"), "%B")
|
assert_equal('May', @filters.date(Time.parse("2006-05-05 10:00:00"), "%B"))
|
||||||
assert_equal 'June', @filters.date(Time.parse("2006-06-05 10:00:00"), "%B")
|
assert_equal('June', @filters.date(Time.parse("2006-06-05 10:00:00"), "%B"))
|
||||||
assert_equal 'July', @filters.date(Time.parse("2006-07-05 10:00:00"), "%B")
|
assert_equal('July', @filters.date(Time.parse("2006-07-05 10:00:00"), "%B"))
|
||||||
|
|
||||||
assert_equal 'May', @filters.date("2006-05-05 10:00:00", "%B")
|
assert_equal('May', @filters.date("2006-05-05 10:00:00", "%B"))
|
||||||
assert_equal 'June', @filters.date("2006-06-05 10:00:00", "%B")
|
assert_equal('June', @filters.date("2006-06-05 10:00:00", "%B"))
|
||||||
assert_equal 'July', @filters.date("2006-07-05 10:00:00", "%B")
|
assert_equal('July', @filters.date("2006-07-05 10:00:00", "%B"))
|
||||||
|
|
||||||
assert_equal '2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", "")
|
assert_equal('2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", ""))
|
||||||
assert_equal '2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", "")
|
assert_equal('2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", ""))
|
||||||
assert_equal '2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", "")
|
assert_equal('2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", ""))
|
||||||
assert_equal '2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", nil)
|
assert_equal('2006-07-05 10:00:00', @filters.date("2006-07-05 10:00:00", nil))
|
||||||
|
|
||||||
assert_equal '07/05/2006', @filters.date("2006-07-05 10:00:00", "%m/%d/%Y")
|
assert_equal('07/05/2006', @filters.date("2006-07-05 10:00:00", "%m/%d/%Y"))
|
||||||
|
|
||||||
assert_equal "07/16/2004", @filters.date("Fri Jul 16 01:00:00 2004", "%m/%d/%Y")
|
assert_equal("07/16/2004", @filters.date("Fri Jul 16 01:00:00 2004", "%m/%d/%Y"))
|
||||||
assert_equal Date.today.year.to_s, @filters.date('now', '%Y')
|
assert_equal(Date.today.year.to_s, @filters.date('now', '%Y'))
|
||||||
assert_equal Date.today.year.to_s, @filters.date('today', '%Y')
|
assert_equal(Date.today.year.to_s, @filters.date('today', '%Y'))
|
||||||
assert_equal Date.today.year.to_s, @filters.date('Today', '%Y')
|
assert_equal(Date.today.year.to_s, @filters.date('Today', '%Y'))
|
||||||
|
|
||||||
assert_nil @filters.date(nil, "%B")
|
assert_nil(@filters.date(nil, "%B"))
|
||||||
|
|
||||||
assert_equal '', @filters.date('', "%B")
|
assert_equal('', @filters.date('', "%B"))
|
||||||
|
|
||||||
with_timezone("UTC") do
|
with_timezone("UTC") do
|
||||||
assert_equal "07/05/2006", @filters.date(1152098955, "%m/%d/%Y")
|
assert_equal("07/05/2006", @filters.date(1152098955, "%m/%d/%Y"))
|
||||||
assert_equal "07/05/2006", @filters.date("1152098955", "%m/%d/%Y")
|
assert_equal("07/05/2006", @filters.date("1152098955", "%m/%d/%Y"))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_first_last
|
def test_first_last
|
||||||
assert_equal 1, @filters.first([1, 2, 3])
|
assert_equal(1, @filters.first([1, 2, 3]))
|
||||||
assert_equal 3, @filters.last([1, 2, 3])
|
assert_equal(3, @filters.last([1, 2, 3]))
|
||||||
assert_nil @filters.first([])
|
assert_nil(@filters.first([]))
|
||||||
assert_nil @filters.last([])
|
assert_nil(@filters.last([]))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_replace
|
def test_replace
|
||||||
assert_equal '2 2 2 2', @filters.replace('1 1 1 1', '1', 2)
|
assert_equal('2 2 2 2', @filters.replace('1 1 1 1', '1', 2))
|
||||||
assert_equal '2 2 2 2', @filters.replace('1 1 1 1', 1, 2)
|
assert_equal('2 2 2 2', @filters.replace('1 1 1 1', 1, 2))
|
||||||
assert_equal '2 1 1 1', @filters.replace_first('1 1 1 1', '1', 2)
|
assert_equal('2 1 1 1', @filters.replace_first('1 1 1 1', '1', 2))
|
||||||
assert_equal '2 1 1 1', @filters.replace_first('1 1 1 1', 1, 2)
|
assert_equal('2 1 1 1', @filters.replace_first('1 1 1 1', 1, 2))
|
||||||
assert_template_result '2 1 1 1', "{{ '1 1 1 1' | replace_first: '1', 2 }}"
|
assert_template_result('2 1 1 1', "{{ '1 1 1 1' | replace_first: '1', 2 }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_remove
|
def test_remove
|
||||||
assert_equal ' ', @filters.remove("a a a a", 'a')
|
assert_equal(' ', @filters.remove("a a a a", 'a'))
|
||||||
assert_equal ' ', @filters.remove("1 1 1 1", 1)
|
assert_equal(' ', @filters.remove("1 1 1 1", 1))
|
||||||
assert_equal 'a a a', @filters.remove_first("a a a a", 'a ')
|
assert_equal('a a a', @filters.remove_first("a a a a", 'a '))
|
||||||
assert_equal ' 1 1 1', @filters.remove_first("1 1 1 1", 1)
|
assert_equal(' 1 1 1', @filters.remove_first("1 1 1 1", 1))
|
||||||
assert_template_result 'a a a', "{{ 'a a a a' | remove_first: 'a ' }}"
|
assert_template_result('a a a', "{{ 'a a a a' | remove_first: 'a ' }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_pipes_in_string_arguments
|
def test_pipes_in_string_arguments
|
||||||
assert_template_result 'foobar', "{{ 'foo|bar' | remove: '|' }}"
|
assert_template_result('foobar', "{{ 'foo|bar' | remove: '|' }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strip
|
def test_strip
|
||||||
assert_template_result 'ab c', "{{ source | strip }}", 'source' => " ab c "
|
assert_template_result('ab c', "{{ source | strip }}", 'source' => " ab c ")
|
||||||
assert_template_result 'ab c', "{{ source | strip }}", 'source' => " \tab c \n \t"
|
assert_template_result('ab c', "{{ source | strip }}", 'source' => " \tab c \n \t")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_lstrip
|
def test_lstrip
|
||||||
assert_template_result 'ab c ', "{{ source | lstrip }}", 'source' => " ab c "
|
assert_template_result('ab c ', "{{ source | lstrip }}", 'source' => " ab c ")
|
||||||
assert_template_result "ab c \n \t", "{{ source | lstrip }}", 'source' => " \tab c \n \t"
|
assert_template_result("ab c \n \t", "{{ source | lstrip }}", 'source' => " \tab c \n \t")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_rstrip
|
def test_rstrip
|
||||||
assert_template_result " ab c", "{{ source | rstrip }}", 'source' => " ab c "
|
assert_template_result(" ab c", "{{ source | rstrip }}", 'source' => " ab c ")
|
||||||
assert_template_result " \tab c", "{{ source | rstrip }}", 'source' => " \tab c \n \t"
|
assert_template_result(" \tab c", "{{ source | rstrip }}", 'source' => " \tab c \n \t")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strip_newlines
|
def test_strip_newlines
|
||||||
assert_template_result 'abc', "{{ source | strip_newlines }}", 'source' => "a\nb\nc"
|
assert_template_result('abc', "{{ source | strip_newlines }}", 'source' => "a\nb\nc")
|
||||||
assert_template_result 'abc', "{{ source | strip_newlines }}", 'source' => "a\r\nb\nc"
|
assert_template_result('abc', "{{ source | strip_newlines }}", 'source' => "a\r\nb\nc")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_newlines_to_br
|
def test_newlines_to_br
|
||||||
assert_template_result "a<br />\nb<br />\nc", "{{ source | newline_to_br }}", 'source' => "a\nb\nc"
|
assert_template_result("a<br />\nb<br />\nc", "{{ source | newline_to_br }}", 'source' => "a\nb\nc")
|
||||||
|
assert_template_result("a<br />\nb<br />\nc", "{{ source | newline_to_br }}", 'source' => "a\r\nb\nc")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_plus
|
def test_plus
|
||||||
assert_template_result "2", "{{ 1 | plus:1 }}"
|
assert_template_result("2", "{{ 1 | plus:1 }}")
|
||||||
assert_template_result "2.0", "{{ '1' | plus:'1.0' }}"
|
assert_template_result("2.0", "{{ '1' | plus:'1.0' }}")
|
||||||
|
|
||||||
assert_template_result "5", "{{ price | plus:'2' }}", 'price' => NumberLikeThing.new(3)
|
assert_template_result("5", "{{ price | plus:'2' }}", 'price' => NumberLikeThing.new(3))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_minus
|
def test_minus
|
||||||
assert_template_result "4", "{{ input | minus:operand }}", 'input' => 5, 'operand' => 1
|
assert_template_result("4", "{{ input | minus:operand }}", 'input' => 5, 'operand' => 1)
|
||||||
assert_template_result "2.3", "{{ '4.3' | minus:'2' }}"
|
assert_template_result("2.3", "{{ '4.3' | minus:'2' }}")
|
||||||
|
|
||||||
assert_template_result "5", "{{ price | minus:'2' }}", 'price' => NumberLikeThing.new(7)
|
assert_template_result("5", "{{ price | minus:'2' }}", 'price' => NumberLikeThing.new(7))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_abs
|
def test_abs
|
||||||
assert_template_result "17", "{{ 17 | abs }}"
|
assert_template_result("17", "{{ 17 | abs }}")
|
||||||
assert_template_result "17", "{{ -17 | abs }}"
|
assert_template_result("17", "{{ -17 | abs }}")
|
||||||
assert_template_result "17", "{{ '17' | abs }}"
|
assert_template_result("17", "{{ '17' | abs }}")
|
||||||
assert_template_result "17", "{{ '-17' | abs }}"
|
assert_template_result("17", "{{ '-17' | abs }}")
|
||||||
assert_template_result "0", "{{ 0 | abs }}"
|
assert_template_result("0", "{{ 0 | abs }}")
|
||||||
assert_template_result "0", "{{ '0' | abs }}"
|
assert_template_result("0", "{{ '0' | abs }}")
|
||||||
assert_template_result "17.42", "{{ 17.42 | abs }}"
|
assert_template_result("17.42", "{{ 17.42 | abs }}")
|
||||||
assert_template_result "17.42", "{{ -17.42 | abs }}"
|
assert_template_result("17.42", "{{ -17.42 | abs }}")
|
||||||
assert_template_result "17.42", "{{ '17.42' | abs }}"
|
assert_template_result("17.42", "{{ '17.42' | abs }}")
|
||||||
assert_template_result "17.42", "{{ '-17.42' | abs }}"
|
assert_template_result("17.42", "{{ '-17.42' | abs }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_times
|
def test_times
|
||||||
assert_template_result "12", "{{ 3 | times:4 }}"
|
assert_template_result("12", "{{ 3 | times:4 }}")
|
||||||
assert_template_result "0", "{{ 'foo' | times:4 }}"
|
assert_template_result("0", "{{ 'foo' | times:4 }}")
|
||||||
assert_template_result "6", "{{ '2.1' | times:3 | replace: '.','-' | plus:0}}"
|
assert_template_result("6", "{{ '2.1' | times:3 | replace: '.','-' | plus:0}}")
|
||||||
assert_template_result "7.25", "{{ 0.0725 | times:100 }}"
|
assert_template_result("7.25", "{{ 0.0725 | times:100 }}")
|
||||||
assert_template_result "-7.25", '{{ "-0.0725" | times:100 }}'
|
assert_template_result("-7.25", '{{ "-0.0725" | times:100 }}')
|
||||||
assert_template_result "7.25", '{{ "-0.0725" | times: -100 }}'
|
assert_template_result("7.25", '{{ "-0.0725" | times: -100 }}')
|
||||||
assert_template_result "4", "{{ price | times:2 }}", 'price' => NumberLikeThing.new(2)
|
assert_template_result("4", "{{ price | times:2 }}", 'price' => NumberLikeThing.new(2))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_divided_by
|
def test_divided_by
|
||||||
assert_template_result "4", "{{ 12 | divided_by:3 }}"
|
assert_template_result("4", "{{ 12 | divided_by:3 }}")
|
||||||
assert_template_result "4", "{{ 14 | divided_by:3 }}"
|
assert_template_result("4", "{{ 14 | divided_by:3 }}")
|
||||||
|
|
||||||
assert_template_result "5", "{{ 15 | divided_by:3 }}"
|
assert_template_result("5", "{{ 15 | divided_by:3 }}")
|
||||||
assert_equal "Liquid error: divided by 0", Template.parse("{{ 5 | divided_by:0 }}").render
|
assert_equal("Liquid error: divided by 0", Template.parse("{{ 5 | divided_by:0 }}").render)
|
||||||
|
|
||||||
assert_template_result "0.5", "{{ 2.0 | divided_by:4 }}"
|
assert_template_result("0.5", "{{ 2.0 | divided_by:4 }}")
|
||||||
assert_raises(Liquid::ZeroDivisionError) do
|
assert_raises(Liquid::ZeroDivisionError) do
|
||||||
assert_template_result "4", "{{ 1 | modulo: 0 }}"
|
assert_template_result("4", "{{ 1 | modulo: 0 }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_template_result "5", "{{ price | divided_by:2 }}", 'price' => NumberLikeThing.new(10)
|
assert_template_result("5", "{{ price | divided_by:2 }}", 'price' => NumberLikeThing.new(10))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_modulo
|
def test_modulo
|
||||||
assert_template_result "1", "{{ 3 | modulo:2 }}"
|
assert_template_result("1", "{{ 3 | modulo:2 }}")
|
||||||
assert_raises(Liquid::ZeroDivisionError) do
|
assert_raises(Liquid::ZeroDivisionError) do
|
||||||
assert_template_result "4", "{{ 1 | modulo: 0 }}"
|
assert_template_result("4", "{{ 1 | modulo: 0 }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_template_result "1", "{{ price | modulo:2 }}", 'price' => NumberLikeThing.new(3)
|
assert_template_result("1", "{{ price | modulo:2 }}", 'price' => NumberLikeThing.new(3))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_round
|
def test_round
|
||||||
assert_template_result "5", "{{ input | round }}", 'input' => 4.6
|
assert_template_result("5", "{{ input | round }}", 'input' => 4.6)
|
||||||
assert_template_result "4", "{{ '4.3' | round }}"
|
assert_template_result("4", "{{ '4.3' | round }}")
|
||||||
assert_template_result "4.56", "{{ input | round: 2 }}", 'input' => 4.5612
|
assert_template_result("4.56", "{{ input | round: 2 }}", 'input' => 4.5612)
|
||||||
assert_raises(Liquid::FloatDomainError) do
|
assert_raises(Liquid::FloatDomainError) do
|
||||||
assert_template_result "4", "{{ 1.0 | divided_by: 0.0 | round }}"
|
assert_template_result("4", "{{ 1.0 | divided_by: 0.0 | round }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_template_result "5", "{{ price | round }}", 'price' => NumberLikeThing.new(4.6)
|
assert_template_result("5", "{{ price | round }}", 'price' => NumberLikeThing.new(4.6))
|
||||||
assert_template_result "4", "{{ price | round }}", 'price' => NumberLikeThing.new(4.3)
|
assert_template_result("4", "{{ price | round }}", 'price' => NumberLikeThing.new(4.3))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_ceil
|
def test_ceil
|
||||||
assert_template_result "5", "{{ input | ceil }}", 'input' => 4.6
|
assert_template_result("5", "{{ input | ceil }}", 'input' => 4.6)
|
||||||
assert_template_result "5", "{{ '4.3' | ceil }}"
|
assert_template_result("5", "{{ '4.3' | ceil }}")
|
||||||
assert_raises(Liquid::FloatDomainError) do
|
assert_raises(Liquid::FloatDomainError) do
|
||||||
assert_template_result "4", "{{ 1.0 | divided_by: 0.0 | ceil }}"
|
assert_template_result("4", "{{ 1.0 | divided_by: 0.0 | ceil }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_template_result "5", "{{ price | ceil }}", 'price' => NumberLikeThing.new(4.6)
|
assert_template_result("5", "{{ price | ceil }}", 'price' => NumberLikeThing.new(4.6))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_floor
|
def test_floor
|
||||||
assert_template_result "4", "{{ input | floor }}", 'input' => 4.6
|
assert_template_result("4", "{{ input | floor }}", 'input' => 4.6)
|
||||||
assert_template_result "4", "{{ '4.3' | floor }}"
|
assert_template_result("4", "{{ '4.3' | floor }}")
|
||||||
assert_raises(Liquid::FloatDomainError) do
|
assert_raises(Liquid::FloatDomainError) do
|
||||||
assert_template_result "4", "{{ 1.0 | divided_by: 0.0 | floor }}"
|
assert_template_result("4", "{{ 1.0 | divided_by: 0.0 | floor }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_template_result "5", "{{ price | floor }}", 'price' => NumberLikeThing.new(5.4)
|
assert_template_result("5", "{{ price | floor }}", 'price' => NumberLikeThing.new(5.4))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_at_most
|
def test_at_most
|
||||||
assert_template_result "4", "{{ 5 | at_most:4 }}"
|
assert_template_result("4", "{{ 5 | at_most:4 }}")
|
||||||
assert_template_result "5", "{{ 5 | at_most:5 }}"
|
assert_template_result("5", "{{ 5 | at_most:5 }}")
|
||||||
assert_template_result "5", "{{ 5 | at_most:6 }}"
|
assert_template_result("5", "{{ 5 | at_most:6 }}")
|
||||||
|
|
||||||
assert_template_result "4.5", "{{ 4.5 | at_most:5 }}"
|
assert_template_result("4.5", "{{ 4.5 | at_most:5 }}")
|
||||||
assert_template_result "5", "{{ width | at_most:5 }}", 'width' => NumberLikeThing.new(6)
|
assert_template_result("5", "{{ width | at_most:5 }}", 'width' => NumberLikeThing.new(6))
|
||||||
assert_template_result "4", "{{ width | at_most:5 }}", 'width' => NumberLikeThing.new(4)
|
assert_template_result("4", "{{ width | at_most:5 }}", 'width' => NumberLikeThing.new(4))
|
||||||
assert_template_result "4", "{{ 5 | at_most: width }}", 'width' => NumberLikeThing.new(4)
|
assert_template_result("4", "{{ 5 | at_most: width }}", 'width' => NumberLikeThing.new(4))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_at_least
|
def test_at_least
|
||||||
assert_template_result "5", "{{ 5 | at_least:4 }}"
|
assert_template_result("5", "{{ 5 | at_least:4 }}")
|
||||||
assert_template_result "5", "{{ 5 | at_least:5 }}"
|
assert_template_result("5", "{{ 5 | at_least:5 }}")
|
||||||
assert_template_result "6", "{{ 5 | at_least:6 }}"
|
assert_template_result("6", "{{ 5 | at_least:6 }}")
|
||||||
|
|
||||||
assert_template_result "5", "{{ 4.5 | at_least:5 }}"
|
assert_template_result("5", "{{ 4.5 | at_least:5 }}")
|
||||||
assert_template_result "6", "{{ width | at_least:5 }}", 'width' => NumberLikeThing.new(6)
|
assert_template_result("6", "{{ width | at_least:5 }}", 'width' => NumberLikeThing.new(6))
|
||||||
assert_template_result "5", "{{ width | at_least:5 }}", 'width' => NumberLikeThing.new(4)
|
assert_template_result("5", "{{ width | at_least:5 }}", 'width' => NumberLikeThing.new(4))
|
||||||
assert_template_result "6", "{{ 5 | at_least: width }}", 'width' => NumberLikeThing.new(6)
|
assert_template_result("6", "{{ 5 | at_least: width }}", 'width' => NumberLikeThing.new(6))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_append
|
def test_append
|
||||||
@@ -660,9 +667,9 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_concat
|
def test_concat
|
||||||
assert_equal [1, 2, 3, 4], @filters.concat([1, 2], [3, 4])
|
assert_equal([1, 2, 3, 4], @filters.concat([1, 2], [3, 4]))
|
||||||
assert_equal [1, 2, 'a'], @filters.concat([1, 2], ['a'])
|
assert_equal([1, 2, 'a'], @filters.concat([1, 2], ['a']))
|
||||||
assert_equal [1, 2, 10], @filters.concat([1, 2], [10])
|
assert_equal([1, 2, 10], @filters.concat([1, 2], [10]))
|
||||||
|
|
||||||
assert_raises(Liquid::ArgumentError, "concat filter requires an array argument") do
|
assert_raises(Liquid::ArgumentError, "concat filter requires an array argument") do
|
||||||
@filters.concat([1, 2], 10)
|
@filters.concat([1, 2], 10)
|
||||||
@@ -676,12 +683,23 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_default
|
def test_default
|
||||||
assert_equal "foo", @filters.default("foo", "bar")
|
assert_equal("foo", @filters.default("foo", "bar"))
|
||||||
assert_equal "bar", @filters.default(nil, "bar")
|
assert_equal("bar", @filters.default(nil, "bar"))
|
||||||
assert_equal "bar", @filters.default("", "bar")
|
assert_equal("bar", @filters.default("", "bar"))
|
||||||
assert_equal "bar", @filters.default(false, "bar")
|
assert_equal("bar", @filters.default(false, "bar"))
|
||||||
assert_equal "bar", @filters.default([], "bar")
|
assert_equal("bar", @filters.default([], "bar"))
|
||||||
assert_equal "bar", @filters.default({}, "bar")
|
assert_equal("bar", @filters.default({}, "bar"))
|
||||||
|
assert_template_result('bar', "{{ false | default: 'bar' }}")
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_default_handle_false
|
||||||
|
assert_equal("foo", @filters.default("foo", "bar", "allow_false" => true))
|
||||||
|
assert_equal("bar", @filters.default(nil, "bar", "allow_false" => true))
|
||||||
|
assert_equal("bar", @filters.default("", "bar", "allow_false" => true))
|
||||||
|
assert_equal(false, @filters.default(false, "bar", "allow_false" => true))
|
||||||
|
assert_equal("bar", @filters.default([], "bar", "allow_false" => true))
|
||||||
|
assert_equal("bar", @filters.default({}, "bar", "allow_false" => true))
|
||||||
|
assert_template_result('false', "{{ false | default: 'bar', allow_false: true }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_cannot_access_private_methods
|
def test_cannot_access_private_methods
|
||||||
@@ -706,8 +724,8 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
{ "handle" => "delta", "ok" => true },
|
{ "handle" => "delta", "ok" => true },
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_equal expectation, @filters.where(input, "ok", true)
|
assert_equal(expectation, @filters.where(input, "ok", true))
|
||||||
assert_equal expectation, @filters.where(input, "ok")
|
assert_equal(expectation, @filters.where(input, "ok"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_where_no_key_set
|
def test_where_no_key_set
|
||||||
@@ -723,13 +741,13 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
{ "handle" => "delta", "ok" => true },
|
{ "handle" => "delta", "ok" => true },
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_equal expectation, @filters.where(input, "ok", true)
|
assert_equal(expectation, @filters.where(input, "ok", true))
|
||||||
assert_equal expectation, @filters.where(input, "ok")
|
assert_equal(expectation, @filters.where(input, "ok"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_where_non_array_map_input
|
def test_where_non_array_map_input
|
||||||
assert_equal [{ "a" => "ok" }], @filters.where({ "a" => "ok" }, "a", "ok")
|
assert_equal([{ "a" => "ok" }], @filters.where({ "a" => "ok" }, "a", "ok"))
|
||||||
assert_equal [], @filters.where({ "a" => "not ok" }, "a", "ok")
|
assert_equal([], @filters.where({ "a" => "not ok" }, "a", "ok"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_where_indexable_but_non_map_value
|
def test_where_indexable_but_non_map_value
|
||||||
@@ -744,14 +762,57 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
{ "message" => "Hallo!", "language" => "German" },
|
{ "message" => "Hallo!", "language" => "German" },
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_equal [{ "message" => "Bonjour!", "language" => "French" }], @filters.where(input, "language", "French")
|
assert_equal([{ "message" => "Bonjour!", "language" => "French" }], @filters.where(input, "language", "French"))
|
||||||
assert_equal [{ "message" => "Hallo!", "language" => "German" }], @filters.where(input, "language", "German")
|
assert_equal([{ "message" => "Hallo!", "language" => "German" }], @filters.where(input, "language", "German"))
|
||||||
assert_equal [{ "message" => "Hello!", "language" => "English" }], @filters.where(input, "language", "English")
|
assert_equal([{ "message" => "Hello!", "language" => "English" }], @filters.where(input, "language", "English"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_where_array_of_only_unindexable_values
|
def test_where_array_of_only_unindexable_values
|
||||||
assert_nil @filters.where([nil], "ok", true)
|
assert_nil(@filters.where([nil], "ok", true))
|
||||||
assert_nil @filters.where([nil], "ok")
|
assert_nil(@filters.where([nil], "ok"))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_all_filters_never_raise_non_liquid_exception
|
||||||
|
test_drop = TestDrop.new
|
||||||
|
test_drop.context = Context.new
|
||||||
|
test_enum = TestEnumerable.new
|
||||||
|
test_enum.context = Context.new
|
||||||
|
test_types = [
|
||||||
|
"foo",
|
||||||
|
123,
|
||||||
|
0,
|
||||||
|
0.0,
|
||||||
|
-1234.003030303,
|
||||||
|
-99999999,
|
||||||
|
1234.38383000383830003838300,
|
||||||
|
nil,
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
TestThing.new,
|
||||||
|
test_drop,
|
||||||
|
test_enum,
|
||||||
|
["foo", "bar"],
|
||||||
|
{ "foo" => "bar" },
|
||||||
|
{ foo: "bar" },
|
||||||
|
[{ "foo" => "bar" }, { "foo" => 123 }, { "foo" => nil }, { "foo" => true }, { "foo" => ["foo", "bar"] }],
|
||||||
|
{ 1 => "bar" },
|
||||||
|
["foo", 123, nil, true, false, Drop, ["foo"], { foo: "bar" }],
|
||||||
|
]
|
||||||
|
test_types.each do |first|
|
||||||
|
test_types.each do |other|
|
||||||
|
(@filters.methods - Object.methods).each do |method|
|
||||||
|
arg_count = @filters.method(method).arity
|
||||||
|
arg_count *= -1 if arg_count < 0
|
||||||
|
inputs = [first]
|
||||||
|
inputs << ([other] * (arg_count - 1)) if arg_count > 1
|
||||||
|
begin
|
||||||
|
@filters.send(method, *inputs)
|
||||||
|
rescue Liquid::ArgumentError, Liquid::ZeroDivisionError
|
||||||
|
nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_where_no_target_value
|
def test_where_no_target_value
|
||||||
@@ -762,7 +823,7 @@ class StandardFiltersTest < Minitest::Test
|
|||||||
{ "bar" => true },
|
{ "bar" => true },
|
||||||
]
|
]
|
||||||
|
|
||||||
assert_equal [{ "foo" => true }, { "foo" => "for sure" }], @filters.where(input, "foo")
|
assert_equal([{ "foo" => true }, { "foo" => "for sure" }], @filters.where(input, "foo"))
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|||||||
59
test/integration/tag/disableable_test.rb
Normal file
59
test/integration/tag/disableable_test.rb
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'test_helper'
|
||||||
|
|
||||||
|
class TagDisableableTest < Minitest::Test
|
||||||
|
include Liquid
|
||||||
|
|
||||||
|
module RenderTagName
|
||||||
|
def render(_context)
|
||||||
|
tag_name
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class Custom < Tag
|
||||||
|
prepend Liquid::Tag::Disableable
|
||||||
|
include RenderTagName
|
||||||
|
end
|
||||||
|
|
||||||
|
class Custom2 < Tag
|
||||||
|
prepend Liquid::Tag::Disableable
|
||||||
|
include RenderTagName
|
||||||
|
end
|
||||||
|
|
||||||
|
class DisableCustom < Block
|
||||||
|
disable_tags "custom"
|
||||||
|
end
|
||||||
|
|
||||||
|
class DisableBoth < Block
|
||||||
|
disable_tags "custom", "custom2"
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_block_tag_disabling_nested_tag
|
||||||
|
with_disableable_tags do
|
||||||
|
with_custom_tag('disable', DisableCustom) do
|
||||||
|
output = Template.parse('{% disable %}{% custom %};{% custom2 %}{% enddisable %}').render
|
||||||
|
assert_equal('Liquid error: custom usage is not allowed in this context;custom2', output)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_block_tag_disabling_multiple_nested_tags
|
||||||
|
with_disableable_tags do
|
||||||
|
with_custom_tag('disable', DisableBoth) do
|
||||||
|
output = Template.parse('{% disable %}{% custom %};{% custom2 %}{% enddisable %}').render
|
||||||
|
assert_equal('Liquid error: custom usage is not allowed in this context;Liquid error: custom2 usage is not allowed in this context', output)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def with_disableable_tags
|
||||||
|
with_custom_tag('custom', Custom) do
|
||||||
|
with_custom_tag('custom2', Custom2) do
|
||||||
|
yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
45
test/integration/tag_test.rb
Normal file
45
test/integration/tag_test.rb
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'test_helper'
|
||||||
|
|
||||||
|
class TagTest < Minitest::Test
|
||||||
|
include Liquid
|
||||||
|
|
||||||
|
def test_custom_tags_have_a_default_render_to_output_buffer_method_for_backwards_compatibility
|
||||||
|
klass1 = Class.new(Tag) do
|
||||||
|
def render(*)
|
||||||
|
'hello'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
with_custom_tag('blabla', klass1) do
|
||||||
|
template = Liquid::Template.parse("{% blabla %}")
|
||||||
|
|
||||||
|
assert_equal('hello', template.render)
|
||||||
|
|
||||||
|
buf = +''
|
||||||
|
output = template.render({}, output: buf)
|
||||||
|
assert_equal('hello', output)
|
||||||
|
assert_equal('hello', buf)
|
||||||
|
assert_equal(buf.object_id, output.object_id)
|
||||||
|
end
|
||||||
|
|
||||||
|
klass2 = Class.new(klass1) do
|
||||||
|
def render(*)
|
||||||
|
'foo' + super + 'bar'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
with_custom_tag('blabla', klass2) do
|
||||||
|
template = Liquid::Template.parse("{% blabla %}")
|
||||||
|
|
||||||
|
assert_equal('foohellobar', template.render)
|
||||||
|
|
||||||
|
buf = +''
|
||||||
|
output = template.render({}, output: buf)
|
||||||
|
assert_equal('foohellobar', output)
|
||||||
|
assert_equal('foohellobar', buf)
|
||||||
|
assert_equal(buf.object_id, output.object_id)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -350,7 +350,7 @@ HERE
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_inner_for_over_empty_input
|
def test_inner_for_over_empty_input
|
||||||
assert_template_result 'oo', '{% for a in (1..2) %}o{% for b in empty %}{% endfor %}{% endfor %}'
|
assert_template_result('oo', '{% for a in (1..2) %}o{% for b in empty %}{% endfor %}{% endfor %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_blank_string_not_iterable
|
def test_blank_string_not_iterable
|
||||||
@@ -394,8 +394,8 @@ HERE
|
|||||||
expected = '12345'
|
expected = '12345'
|
||||||
template = '{% for item in items %}{{item}}{% endfor %}'
|
template = '{% for item in items %}{{item}}{% endfor %}'
|
||||||
assert_template_result(expected, template, assigns)
|
assert_template_result(expected, template, assigns)
|
||||||
assert loader.each_called
|
assert(loader.each_called)
|
||||||
assert !loader.load_slice_called
|
assert(!loader.load_slice_called)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_iterate_with_load_slice_when_limit_applied
|
def test_iterate_with_load_slice_when_limit_applied
|
||||||
@@ -404,8 +404,8 @@ HERE
|
|||||||
expected = '1'
|
expected = '1'
|
||||||
template = '{% for item in items limit:1 %}{{item}}{% endfor %}'
|
template = '{% for item in items limit:1 %}{{item}}{% endfor %}'
|
||||||
assert_template_result(expected, template, assigns)
|
assert_template_result(expected, template, assigns)
|
||||||
assert !loader.each_called
|
assert(!loader.each_called)
|
||||||
assert loader.load_slice_called
|
assert(loader.load_slice_called)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_iterate_with_load_slice_when_limit_and_offset_applied
|
def test_iterate_with_load_slice_when_limit_and_offset_applied
|
||||||
@@ -414,8 +414,8 @@ HERE
|
|||||||
expected = '34'
|
expected = '34'
|
||||||
template = '{% for item in items offset:2 limit:2 %}{{item}}{% endfor %}'
|
template = '{% for item in items offset:2 limit:2 %}{{item}}{% endfor %}'
|
||||||
assert_template_result(expected, template, assigns)
|
assert_template_result(expected, template, assigns)
|
||||||
assert !loader.each_called
|
assert(!loader.each_called)
|
||||||
assert loader.load_slice_called
|
assert(loader.load_slice_called)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_iterate_with_load_slice_returns_same_results_as_without
|
def test_iterate_with_load_slice_returns_same_results_as_without
|
||||||
@@ -435,6 +435,32 @@ HERE
|
|||||||
Liquid::Template.parse('{% for i in (1..2) %}{{ standard_error }}{% endfor %}').render!(context)
|
Liquid::Template.parse('{% for i in (1..2) %}{{ standard_error }}{% endfor %}').render!(context)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert context.registers[:for_stack].empty?
|
assert(context.registers[:for_stack].empty?)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_instrument_for_offset_continue
|
||||||
|
assert_usage_increment('for_offset_continue') do
|
||||||
|
Template.parse('{% for item in items offset:continue %}{{item}}{% endfor %}')
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_usage_increment('for_offset_continue', times: 0) do
|
||||||
|
Template.parse('{% for item in items offset:2 %}{{item}}{% endfor %}')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_instrument_forloop_drop_name
|
||||||
|
assigns = { 'items' => [1, 2, 3, 4, 5] }
|
||||||
|
|
||||||
|
assert_usage_increment('forloop_drop_name', times: 5) do
|
||||||
|
Template.parse('{% for item in items %}{{forloop.name}}{% endfor %}').render!(assigns)
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_usage_increment('forloop_drop_name', times: 0) do
|
||||||
|
Template.parse('{% for item in items %}{{forloop.index}}{% endfor %}').render!(assigns)
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_usage_increment('forloop_drop_name', times: 0) do
|
||||||
|
Template.parse('{% for item in items %}{{item}}{% endfor %}').render!(assigns)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -184,7 +184,7 @@ class IfElseTagTest < Minitest::Test
|
|||||||
tests.each do |vals, expected|
|
tests.each do |vals, expected|
|
||||||
a, b, c = vals
|
a, b, c = vals
|
||||||
assigns = { 'a' => a, 'b' => b, 'c' => c }
|
assigns = { 'a' => a, 'b' => b, 'c' => c }
|
||||||
assert_template_result expected.to_s, tpl, assigns, assigns.to_s
|
assert_template_result(expected.to_s, tpl, assigns, assigns.to_s)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -8,6 +8,9 @@ class TestFileSystem
|
|||||||
when "product"
|
when "product"
|
||||||
"Product: {{ product.title }} "
|
"Product: {{ product.title }} "
|
||||||
|
|
||||||
|
when "product_alias"
|
||||||
|
"Product: {{ product.title }} "
|
||||||
|
|
||||||
when "locale_variables"
|
when "locale_variables"
|
||||||
"Locale: {{echo1}} {{echo2}}"
|
"Locale: {{echo1}} {{echo2}}"
|
||||||
|
|
||||||
@@ -82,56 +85,66 @@ class IncludeTagTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_looks_for_file_system_in_registers_first
|
def test_include_tag_looks_for_file_system_in_registers_first
|
||||||
assert_equal 'from OtherFileSystem',
|
assert_equal('from OtherFileSystem',
|
||||||
Template.parse("{% include 'pick_a_source' %}").render!({}, registers: { file_system: OtherFileSystem.new })
|
Template.parse("{% include 'pick_a_source' %}").render!({}, registers: { file_system: OtherFileSystem.new }))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_with
|
def test_include_tag_with
|
||||||
assert_template_result "Product: Draft 151cm ",
|
assert_template_result("Product: Draft 151cm ",
|
||||||
"{% include 'product' with products[0] %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }]
|
"{% include 'product' with products[0] %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_include_tag_with_alias
|
||||||
|
assert_template_result("Product: Draft 151cm ",
|
||||||
|
"{% include 'product_alias' with products[0] as product %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_include_tag_for_alias
|
||||||
|
assert_template_result("Product: Draft 151cm Product: Element 155cm ",
|
||||||
|
"{% include 'product_alias' for products as product %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_with_default_name
|
def test_include_tag_with_default_name
|
||||||
assert_template_result "Product: Draft 151cm ",
|
assert_template_result("Product: Draft 151cm ",
|
||||||
"{% include 'product' %}", "product" => { 'title' => 'Draft 151cm' }
|
"{% include 'product' %}", "product" => { 'title' => 'Draft 151cm' })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_for
|
def test_include_tag_for
|
||||||
assert_template_result "Product: Draft 151cm Product: Element 155cm ",
|
assert_template_result("Product: Draft 151cm Product: Element 155cm ",
|
||||||
"{% include 'product' for products %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }]
|
"{% include 'product' for products %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_with_local_variables
|
def test_include_tag_with_local_variables
|
||||||
assert_template_result "Locale: test123 ", "{% include 'locale_variables' echo1: 'test123' %}"
|
assert_template_result("Locale: test123 ", "{% include 'locale_variables' echo1: 'test123' %}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_with_multiple_local_variables
|
def test_include_tag_with_multiple_local_variables
|
||||||
assert_template_result "Locale: test123 test321",
|
assert_template_result("Locale: test123 test321",
|
||||||
"{% include 'locale_variables' echo1: 'test123', echo2: 'test321' %}"
|
"{% include 'locale_variables' echo1: 'test123', echo2: 'test321' %}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_with_multiple_local_variables_from_context
|
def test_include_tag_with_multiple_local_variables_from_context
|
||||||
assert_template_result "Locale: test123 test321",
|
assert_template_result("Locale: test123 test321",
|
||||||
"{% include 'locale_variables' echo1: echo1, echo2: more_echos.echo2 %}",
|
"{% include 'locale_variables' echo1: echo1, echo2: more_echos.echo2 %}",
|
||||||
'echo1' => 'test123', 'more_echos' => { "echo2" => 'test321' }
|
'echo1' => 'test123', 'more_echos' => { "echo2" => 'test321' })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_included_templates_assigns_variables
|
def test_included_templates_assigns_variables
|
||||||
assert_template_result "bar", "{% include 'assignments' %}{{ foo }}"
|
assert_template_result("bar", "{% include 'assignments' %}{{ foo }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nested_include_tag
|
def test_nested_include_tag
|
||||||
assert_template_result "body body_detail", "{% include 'body' %}"
|
assert_template_result("body body_detail", "{% include 'body' %}")
|
||||||
|
|
||||||
assert_template_result "header body body_detail footer", "{% include 'nested_template' %}"
|
assert_template_result("header body body_detail footer", "{% include 'nested_template' %}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nested_include_with_variable
|
def test_nested_include_with_variable
|
||||||
assert_template_result "Product: Draft 151cm details ",
|
assert_template_result("Product: Draft 151cm details ",
|
||||||
"{% include 'nested_product_template' with product %}", "product" => { "title" => 'Draft 151cm' }
|
"{% include 'nested_product_template' with product %}", "product" => { "title" => 'Draft 151cm' })
|
||||||
|
|
||||||
assert_template_result "Product: Draft 151cm details Product: Element 155cm details ",
|
assert_template_result("Product: Draft 151cm details Product: Element 155cm details ",
|
||||||
"{% include 'nested_product_template' for products %}", "products" => [{ "title" => 'Draft 151cm' }, { "title" => 'Element 155cm' }]
|
"{% include 'nested_product_template' for products %}", "products" => [{ "title" => 'Draft 151cm' }, { "title" => 'Element 155cm' }])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_recursively_included_template_does_not_produce_endless_loop
|
def test_recursively_included_template_does_not_produce_endless_loop
|
||||||
@@ -149,41 +162,41 @@ class IncludeTagTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_dynamically_choosen_template
|
def test_dynamically_choosen_template
|
||||||
assert_template_result "Test123", "{% include template %}", "template" => 'Test123'
|
assert_template_result("Test123", "{% include template %}", "template" => 'Test123')
|
||||||
assert_template_result "Test321", "{% include template %}", "template" => 'Test321'
|
assert_template_result("Test321", "{% include template %}", "template" => 'Test321')
|
||||||
|
|
||||||
assert_template_result "Product: Draft 151cm ", "{% include template for product %}",
|
assert_template_result("Product: Draft 151cm ", "{% include template for product %}",
|
||||||
"template" => 'product', 'product' => { 'title' => 'Draft 151cm' }
|
"template" => 'product', 'product' => { 'title' => 'Draft 151cm' })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_caches_second_read_of_same_partial
|
def test_include_tag_caches_second_read_of_same_partial
|
||||||
file_system = CountingFileSystem.new
|
file_system = CountingFileSystem.new
|
||||||
assert_equal 'from CountingFileSystemfrom CountingFileSystem',
|
assert_equal('from CountingFileSystemfrom CountingFileSystem',
|
||||||
Template.parse("{% include 'pick_a_source' %}{% include 'pick_a_source' %}").render!({}, registers: { file_system: file_system })
|
Template.parse("{% include 'pick_a_source' %}{% include 'pick_a_source' %}").render!({}, registers: { file_system: file_system }))
|
||||||
assert_equal 1, file_system.count
|
assert_equal(1, file_system.count)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_doesnt_cache_partials_across_renders
|
def test_include_tag_doesnt_cache_partials_across_renders
|
||||||
file_system = CountingFileSystem.new
|
file_system = CountingFileSystem.new
|
||||||
assert_equal 'from CountingFileSystem',
|
assert_equal('from CountingFileSystem',
|
||||||
Template.parse("{% include 'pick_a_source' %}").render!({}, registers: { file_system: file_system })
|
Template.parse("{% include 'pick_a_source' %}").render!({}, registers: { file_system: file_system }))
|
||||||
assert_equal 1, file_system.count
|
assert_equal(1, file_system.count)
|
||||||
|
|
||||||
assert_equal 'from CountingFileSystem',
|
assert_equal('from CountingFileSystem',
|
||||||
Template.parse("{% include 'pick_a_source' %}").render!({}, registers: { file_system: file_system })
|
Template.parse("{% include 'pick_a_source' %}").render!({}, registers: { file_system: file_system }))
|
||||||
assert_equal 2, file_system.count
|
assert_equal(2, file_system.count)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_within_if_statement
|
def test_include_tag_within_if_statement
|
||||||
assert_template_result "foo_if_true", "{% if true %}{% include 'foo_if_true' %}{% endif %}"
|
assert_template_result("foo_if_true", "{% if true %}{% include 'foo_if_true' %}{% endif %}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_custom_include_tag
|
def test_custom_include_tag
|
||||||
original_tag = Liquid::Template.tags['include']
|
original_tag = Liquid::Template.tags['include']
|
||||||
Liquid::Template.tags['include'] = CustomInclude
|
Liquid::Template.tags['include'] = CustomInclude
|
||||||
begin
|
begin
|
||||||
assert_equal "custom_foo",
|
assert_equal("custom_foo",
|
||||||
Template.parse("{% include 'custom_foo' %}").render!
|
Template.parse("{% include 'custom_foo' %}").render!)
|
||||||
ensure
|
ensure
|
||||||
Liquid::Template.tags['include'] = original_tag
|
Liquid::Template.tags['include'] = original_tag
|
||||||
end
|
end
|
||||||
@@ -193,8 +206,8 @@ class IncludeTagTest < Minitest::Test
|
|||||||
original_tag = Liquid::Template.tags['include']
|
original_tag = Liquid::Template.tags['include']
|
||||||
Liquid::Template.tags['include'] = CustomInclude
|
Liquid::Template.tags['include'] = CustomInclude
|
||||||
begin
|
begin
|
||||||
assert_equal "custom_foo_if_true",
|
assert_equal("custom_foo_if_true",
|
||||||
Template.parse("{% if true %}{% include 'custom_foo_if_true' %}{% endif %}").render!
|
Template.parse("{% if true %}{% include 'custom_foo_if_true' %}{% endif %}").render!)
|
||||||
ensure
|
ensure
|
||||||
Liquid::Template.tags['include'] = original_tag
|
Liquid::Template.tags['include'] = original_tag
|
||||||
end
|
end
|
||||||
@@ -205,7 +218,7 @@ class IncludeTagTest < Minitest::Test
|
|||||||
|
|
||||||
a = Liquid::Template.parse(' {% include "nested_template" %}')
|
a = Liquid::Template.parse(' {% include "nested_template" %}')
|
||||||
a.render!
|
a.render!
|
||||||
assert_empty a.errors
|
assert_empty(a.errors)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_passing_options_to_included_templates
|
def test_passing_options_to_included_templates
|
||||||
@@ -213,13 +226,13 @@ class IncludeTagTest < Minitest::Test
|
|||||||
Template.parse("{% include template %}", error_mode: :strict).render!("template" => '{{ "X" || downcase }}')
|
Template.parse("{% include template %}", error_mode: :strict).render!("template" => '{{ "X" || downcase }}')
|
||||||
end
|
end
|
||||||
with_error_mode(:lax) do
|
with_error_mode(:lax) do
|
||||||
assert_equal 'x', Template.parse("{% include template %}", error_mode: :strict, include_options_blacklist: true).render!("template" => '{{ "X" || downcase }}')
|
assert_equal('x', Template.parse("{% include template %}", error_mode: :strict, include_options_blacklist: true).render!("template" => '{{ "X" || downcase }}'))
|
||||||
end
|
end
|
||||||
assert_raises(Liquid::SyntaxError) do
|
assert_raises(Liquid::SyntaxError) do
|
||||||
Template.parse("{% include template %}", error_mode: :strict, include_options_blacklist: [:locale]).render!("template" => '{{ "X" || downcase }}')
|
Template.parse("{% include template %}", error_mode: :strict, include_options_blacklist: [:locale]).render!("template" => '{{ "X" || downcase }}')
|
||||||
end
|
end
|
||||||
with_error_mode(:lax) do
|
with_error_mode(:lax) do
|
||||||
assert_equal 'x', Template.parse("{% include template %}", error_mode: :strict, include_options_blacklist: [:error_mode]).render!("template" => '{{ "X" || downcase }}')
|
assert_equal('x', Template.parse("{% include template %}", error_mode: :strict, include_options_blacklist: [:error_mode]).render!("template" => '{{ "X" || downcase }}'))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -235,22 +248,22 @@ class IncludeTagTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_including_via_variable_value
|
def test_including_via_variable_value
|
||||||
assert_template_result "from TestFileSystem", "{% assign page = 'pick_a_source' %}{% include page %}"
|
assert_template_result("from TestFileSystem", "{% assign page = 'pick_a_source' %}{% include page %}")
|
||||||
|
|
||||||
assert_template_result "Product: Draft 151cm ", "{% assign page = 'product' %}{% include page %}", "product" => { 'title' => 'Draft 151cm' }
|
assert_template_result("Product: Draft 151cm ", "{% assign page = 'product' %}{% include page %}", "product" => { 'title' => 'Draft 151cm' })
|
||||||
|
|
||||||
assert_template_result "Product: Draft 151cm ", "{% assign page = 'product' %}{% include page for foo %}", "foo" => { 'title' => 'Draft 151cm' }
|
assert_template_result("Product: Draft 151cm ", "{% assign page = 'product' %}{% include page for foo %}", "foo" => { 'title' => 'Draft 151cm' })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_including_with_strict_variables
|
def test_including_with_strict_variables
|
||||||
template = Liquid::Template.parse("{% include 'simple' %}", error_mode: :warn)
|
template = Liquid::Template.parse("{% include 'simple' %}", error_mode: :warn)
|
||||||
template.render(nil, strict_variables: true)
|
template.render(nil, strict_variables: true)
|
||||||
|
|
||||||
assert_equal [], template.errors
|
assert_equal([], template.errors)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_break_through_include
|
def test_break_through_include
|
||||||
assert_template_result "1", "{% for i in (1..3) %}{{ i }}{% break %}{{ i }}{% endfor %}"
|
assert_template_result("1", "{% for i in (1..3) %}{{ i }}{% break %}{{ i }}{% endfor %}")
|
||||||
assert_template_result "1", "{% for i in (1..3) %}{{ i }}{% include 'break' %}{{ i }}{% endfor %}"
|
assert_template_result("1", "{% for i in (1..3) %}{{ i }}{% include 'break' %}{{ i }}{% endfor %}")
|
||||||
end
|
end
|
||||||
end # IncludeTagTest
|
end # IncludeTagTest
|
||||||
|
|||||||
59
test/integration/tags/inline_comment_test.rb
Normal file
59
test/integration/tags/inline_comment_test.rb
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'test_helper'
|
||||||
|
|
||||||
|
class InlineCommentTest < Minitest::Test
|
||||||
|
include Liquid
|
||||||
|
|
||||||
|
def test_tag_in_different_styles
|
||||||
|
assert_template_result('', '{% # This text gets ignored %}')
|
||||||
|
assert_template_result('', '{%# This text gets ignored #%}')
|
||||||
|
assert_template_result('', '{%# This text gets ignored %}')
|
||||||
|
assert_template_result('', '{%#- This text gets ignored -#%}')
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_test_syntax_error
|
||||||
|
assert_template_result('fail', '{% #This doesnt work %}')
|
||||||
|
|
||||||
|
assert false
|
||||||
|
rescue
|
||||||
|
# ok good
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_tag_ws_stripping
|
||||||
|
assert_template_result('', ' {%#- This text gets ignored -#%} ')
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_comment_inline_tag
|
||||||
|
assert_template_result('ok', '{% echo "ok" # output something from a tag %}')
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_comment_line_before_tag
|
||||||
|
assert_template_result('ok', '{% # this sort of comment also
|
||||||
|
echo "ok" %}')
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_comment_inline_variable
|
||||||
|
assert_template_result('ok', '{{ "ok" # output something from a variable }}')
|
||||||
|
assert_template_result('ok', '{{ "OK" | downcase # output something from a variable }}')
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_inside_liquid_tag
|
||||||
|
source = <<~LIQUID
|
||||||
|
{%- liquid
|
||||||
|
echo "before("
|
||||||
|
# This text gets ignored
|
||||||
|
echo ")after"
|
||||||
|
-%}
|
||||||
|
LIQUID
|
||||||
|
assert_template_result('before()after', source)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_multiline
|
||||||
|
assert_template_result('', '{% # this sort of comment also
|
||||||
|
# will just work, because it parses
|
||||||
|
# as a single call to the "#" tag %}')
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
@@ -81,6 +81,16 @@ class LiquidTagTest < Minitest::Test
|
|||||||
assert_match_syntax_error("syntax error (line 3): Unknown tag 'error'", "{% liquid echo ''\n \n error %}")
|
assert_match_syntax_error("syntax error (line 3): Unknown tag 'error'", "{% liquid echo ''\n \n error %}")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_nested_liquid_tag
|
||||||
|
assert_template_result('good', <<~LIQUID)
|
||||||
|
{%- if true %}
|
||||||
|
{%- liquid
|
||||||
|
echo "good"
|
||||||
|
%}
|
||||||
|
{%- endif -%}
|
||||||
|
LIQUID
|
||||||
|
end
|
||||||
|
|
||||||
def test_cannot_open_blocks_living_past_a_liquid_tag
|
def test_cannot_open_blocks_living_past_a_liquid_tag
|
||||||
assert_match_syntax_error("syntax error (line 3): 'if' tag was never closed", <<~LIQUID)
|
assert_match_syntax_error("syntax error (line 3): 'if' tag was never closed", <<~LIQUID)
|
||||||
{%- liquid
|
{%- liquid
|
||||||
@@ -90,8 +100,8 @@ class LiquidTagTest < Minitest::Test
|
|||||||
LIQUID
|
LIQUID
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_quirk_can_close_blocks_created_before_a_liquid_tag
|
def test_cannot_close_blocks_created_before_a_liquid_tag
|
||||||
assert_template_result("42", <<~LIQUID)
|
assert_match_syntax_error("syntax error (line 3): 'endif' is not a valid delimiter for liquid tags. use %}", <<~LIQUID)
|
||||||
{%- if true -%}
|
{%- if true -%}
|
||||||
42
|
42
|
||||||
{%- liquid endif -%}
|
{%- liquid endif -%}
|
||||||
|
|||||||
@@ -6,23 +6,24 @@ class RawTagTest < Minitest::Test
|
|||||||
include Liquid
|
include Liquid
|
||||||
|
|
||||||
def test_tag_in_raw
|
def test_tag_in_raw
|
||||||
assert_template_result '{% comment %} test {% endcomment %}',
|
assert_template_result('{% comment %} test {% endcomment %}',
|
||||||
'{% raw %}{% comment %} test {% endcomment %}{% endraw %}'
|
'{% raw %}{% comment %} test {% endcomment %}{% endraw %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_output_in_raw
|
def test_output_in_raw
|
||||||
assert_template_result '{{ test }}', '{% raw %}{{ test }}{% endraw %}'
|
assert_template_result('{{ test }}', '{% raw %}{{ test }}{% endraw %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_open_tag_in_raw
|
def test_open_tag_in_raw
|
||||||
assert_template_result ' Foobar {% invalid ', '{% raw %} Foobar {% invalid {% endraw %}'
|
assert_template_result(' Foobar {% invalid ', '{% raw %} Foobar {% invalid {% endraw %}')
|
||||||
assert_template_result ' Foobar invalid %} ', '{% raw %} Foobar invalid %} {% endraw %}'
|
assert_template_result(' Foobar invalid %} ', '{% raw %} Foobar invalid %} {% endraw %}')
|
||||||
assert_template_result ' Foobar {{ invalid ', '{% raw %} Foobar {{ invalid {% endraw %}'
|
assert_template_result(' Foobar {{ invalid ', '{% raw %} Foobar {{ invalid {% endraw %}')
|
||||||
assert_template_result ' Foobar invalid }} ', '{% raw %} Foobar invalid }} {% endraw %}'
|
assert_template_result(' Foobar invalid }} ', '{% raw %} Foobar invalid }} {% endraw %}')
|
||||||
assert_template_result ' Foobar {% invalid {% {% endraw ', '{% raw %} Foobar {% invalid {% {% endraw {% endraw %}'
|
assert_template_result(' Foobar {% invalid {% {% endraw ', '{% raw %} Foobar {% invalid {% {% endraw {% endraw %}')
|
||||||
assert_template_result ' Foobar {% {% {% ', '{% raw %} Foobar {% {% {% {% endraw %}'
|
assert_template_result(' Foobar {% {% {% ', '{% raw %} Foobar {% {% {% {% endraw %}')
|
||||||
assert_template_result ' test {% raw %} {% endraw %}', '{% raw %} test {% raw %} {% {% endraw %}endraw %}'
|
assert_template_result(' test {% raw %} {% endraw %}', '{% raw %} test {% raw %} {% {% endraw %}endraw %}')
|
||||||
assert_template_result ' Foobar {{ invalid 1', '{% raw %} Foobar {{ invalid {% endraw %}{{ 1 }}'
|
assert_template_result(' Foobar {{ invalid 1', '{% raw %} Foobar {{ invalid {% endraw %}{{ 1 }}')
|
||||||
|
assert_template_result(' Foobar {% foo {% bar %}', '{% raw %} Foobar {% foo {% bar %}{% endraw %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_invalid_raw
|
def test_invalid_raw
|
||||||
|
|||||||
@@ -7,70 +7,44 @@ class RenderTagTest < Minitest::Test
|
|||||||
|
|
||||||
def test_render_with_no_arguments
|
def test_render_with_no_arguments
|
||||||
Liquid::Template.file_system = StubFileSystem.new('source' => 'rendered content')
|
Liquid::Template.file_system = StubFileSystem.new('source' => 'rendered content')
|
||||||
assert_template_result 'rendered content', '{% render "source" %}'
|
assert_template_result('rendered content', '{% render "source" %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_tag_looks_for_file_system_in_registers_first
|
def test_render_tag_looks_for_file_system_in_registers_first
|
||||||
file_system = StubFileSystem.new('pick_a_source' => 'from register file system')
|
file_system = StubFileSystem.new('pick_a_source' => 'from register file system')
|
||||||
assert_equal 'from register file system',
|
assert_equal('from register file system',
|
||||||
Template.parse('{% render "pick_a_source" %}').render!({}, registers: { file_system: file_system })
|
Template.parse('{% render "pick_a_source" %}').render!({}, registers: { file_system: file_system }))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_passes_named_arguments_into_inner_scope
|
def test_render_passes_named_arguments_into_inner_scope
|
||||||
Liquid::Template.file_system = StubFileSystem.new('product' => '{{ inner_product.title }}')
|
Liquid::Template.file_system = StubFileSystem.new('product' => '{{ inner_product.title }}')
|
||||||
assert_template_result 'My Product', '{% render "product", inner_product: outer_product %}',
|
assert_template_result('My Product', '{% render "product", inner_product: outer_product %}',
|
||||||
'outer_product' => { 'title' => 'My Product' }
|
'outer_product' => { 'title' => 'My Product' })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_accepts_literals_as_arguments
|
def test_render_accepts_literals_as_arguments
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ price }}')
|
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ price }}')
|
||||||
assert_template_result '123', '{% render "snippet", price: 123 %}'
|
assert_template_result('123', '{% render "snippet", price: 123 %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_accepts_multiple_named_arguments
|
def test_render_accepts_multiple_named_arguments
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ one }} {{ two }}')
|
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ one }} {{ two }}')
|
||||||
assert_template_result '1 2', '{% render "snippet", one: 1, two: 2 %}'
|
assert_template_result('1 2', '{% render "snippet", one: 1, two: 2 %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_does_not_inherit_parent_scope_variables
|
def test_render_does_not_inherit_parent_scope_variables
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ outer_variable }}')
|
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ outer_variable }}')
|
||||||
assert_template_result '', '{% assign outer_variable = "should not be visible" %}{% render "snippet" %}'
|
assert_template_result('', '{% assign outer_variable = "should not be visible" %}{% render "snippet" %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_does_not_inherit_variable_with_same_name_as_snippet
|
def test_render_does_not_inherit_variable_with_same_name_as_snippet
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ snippet }}')
|
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ snippet }}')
|
||||||
assert_template_result '', "{% assign snippet = 'should not be visible' %}{% render 'snippet' %}"
|
assert_template_result('', "{% assign snippet = 'should not be visible' %}{% render 'snippet' %}")
|
||||||
end
|
|
||||||
|
|
||||||
def test_render_sets_the_correct_template_name_for_errors
|
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ unsafe }}')
|
|
||||||
|
|
||||||
with_taint_mode :error do
|
|
||||||
template = Liquid::Template.parse('{% render "snippet", unsafe: unsafe %}')
|
|
||||||
context = Context.new('unsafe' => (+'unsafe').tap(&:taint))
|
|
||||||
template.render(context)
|
|
||||||
|
|
||||||
assert_equal [Liquid::TaintedError], template.errors.map(&:class)
|
|
||||||
assert_equal 'snippet', template.errors.first.template_name
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_render_sets_the_correct_template_name_for_warnings
|
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{{ unsafe }}')
|
|
||||||
|
|
||||||
with_taint_mode :warn do
|
|
||||||
template = Liquid::Template.parse('{% render "snippet", unsafe: unsafe %}')
|
|
||||||
context = Context.new('unsafe' => (+'unsafe').tap(&:taint))
|
|
||||||
template.render(context)
|
|
||||||
|
|
||||||
assert_equal [Liquid::TaintedError], context.warnings.map(&:class)
|
|
||||||
assert_equal 'snippet', context.warnings.first.template_name
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_does_not_mutate_parent_scope
|
def test_render_does_not_mutate_parent_scope
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{% assign inner = 1 %}')
|
Liquid::Template.file_system = StubFileSystem.new('snippet' => '{% assign inner = 1 %}')
|
||||||
assert_template_result '', "{% render 'snippet' %}{{ inner }}"
|
assert_template_result('', "{% render 'snippet' %}{{ inner }}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nested_render_tag
|
def test_nested_render_tag
|
||||||
@@ -78,13 +52,13 @@ class RenderTagTest < Minitest::Test
|
|||||||
'one' => "one {% render 'two' %}",
|
'one' => "one {% render 'two' %}",
|
||||||
'two' => 'two'
|
'two' => 'two'
|
||||||
)
|
)
|
||||||
assert_template_result 'one two', "{% render 'one' %}"
|
assert_template_result('one two', "{% render 'one' %}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_recursively_rendered_template_does_not_produce_endless_loop
|
def test_recursively_rendered_template_does_not_produce_endless_loop
|
||||||
Liquid::Template.file_system = StubFileSystem.new('loop' => '{% render "loop" %}')
|
Liquid::Template.file_system = StubFileSystem.new('loop' => '{% render "loop" %}')
|
||||||
|
|
||||||
assert_raises Liquid::StackLevelError do
|
assert_raises(Liquid::StackLevelError) do
|
||||||
Template.parse('{% render "loop" %}').render!
|
Template.parse('{% render "loop" %}').render!
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -93,7 +67,7 @@ class RenderTagTest < Minitest::Test
|
|||||||
Liquid::Template.file_system = StubFileSystem.new(
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
'loop_render' => '{% render "loop_render" %}',
|
'loop_render' => '{% render "loop_render" %}',
|
||||||
)
|
)
|
||||||
assert_raises Liquid::StackLevelError do
|
assert_raises(Liquid::StackLevelError) do
|
||||||
Template.parse('{% render "loop_render" %}').render!
|
Template.parse('{% render "loop_render" %}').render!
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -101,50 +75,50 @@ class RenderTagTest < Minitest::Test
|
|||||||
def test_dynamically_choosen_templates_are_not_allowed
|
def test_dynamically_choosen_templates_are_not_allowed
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => 'should not be rendered')
|
Liquid::Template.file_system = StubFileSystem.new('snippet' => 'should not be rendered')
|
||||||
|
|
||||||
assert_raises Liquid::SyntaxError do
|
assert_raises(Liquid::SyntaxError) do
|
||||||
Liquid::Template.parse("{% assign name = 'snippet' %}{% render name %}")
|
Liquid::Template.parse("{% assign name = 'snippet' %}{% render name %}")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_include_tag_caches_second_read_of_same_partial
|
def test_include_tag_caches_second_read_of_same_partial
|
||||||
file_system = StubFileSystem.new('snippet' => 'echo')
|
file_system = StubFileSystem.new('snippet' => 'echo')
|
||||||
assert_equal 'echoecho',
|
assert_equal('echoecho',
|
||||||
Template.parse('{% render "snippet" %}{% render "snippet" %}')
|
Template.parse('{% render "snippet" %}{% render "snippet" %}')
|
||||||
.render!({}, registers: { file_system: file_system })
|
.render!({}, registers: { file_system: file_system }))
|
||||||
assert_equal 1, file_system.file_read_count
|
assert_equal(1, file_system.file_read_count)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_tag_doesnt_cache_partials_across_renders
|
def test_render_tag_doesnt_cache_partials_across_renders
|
||||||
file_system = StubFileSystem.new('snippet' => 'my message')
|
file_system = StubFileSystem.new('snippet' => 'my message')
|
||||||
|
|
||||||
assert_equal 'my message',
|
assert_equal('my message',
|
||||||
Template.parse('{% include "snippet" %}').render!({}, registers: { file_system: file_system })
|
Template.parse('{% include "snippet" %}').render!({}, registers: { file_system: file_system }))
|
||||||
assert_equal 1, file_system.file_read_count
|
assert_equal(1, file_system.file_read_count)
|
||||||
|
|
||||||
assert_equal 'my message',
|
assert_equal('my message',
|
||||||
Template.parse('{% include "snippet" %}').render!({}, registers: { file_system: file_system })
|
Template.parse('{% include "snippet" %}').render!({}, registers: { file_system: file_system }))
|
||||||
assert_equal 2, file_system.file_read_count
|
assert_equal(2, file_system.file_read_count)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_tag_within_if_statement
|
def test_render_tag_within_if_statement
|
||||||
Liquid::Template.file_system = StubFileSystem.new('snippet' => 'my message')
|
Liquid::Template.file_system = StubFileSystem.new('snippet' => 'my message')
|
||||||
assert_template_result 'my message', '{% if true %}{% render "snippet" %}{% endif %}'
|
assert_template_result('my message', '{% if true %}{% render "snippet" %}{% endif %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_break_through_render
|
def test_break_through_render
|
||||||
Liquid::Template.file_system = StubFileSystem.new('break' => '{% break %}')
|
Liquid::Template.file_system = StubFileSystem.new('break' => '{% break %}')
|
||||||
assert_template_result '1', '{% for i in (1..3) %}{{ i }}{% break %}{{ i }}{% endfor %}'
|
assert_template_result('1', '{% for i in (1..3) %}{{ i }}{% break %}{{ i }}{% endfor %}')
|
||||||
assert_template_result '112233', '{% for i in (1..3) %}{{ i }}{% render "break" %}{{ i }}{% endfor %}'
|
assert_template_result('112233', '{% for i in (1..3) %}{{ i }}{% render "break" %}{{ i }}{% endfor %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_increment_is_isolated_between_renders
|
def test_increment_is_isolated_between_renders
|
||||||
Liquid::Template.file_system = StubFileSystem.new('incr' => '{% increment %}')
|
Liquid::Template.file_system = StubFileSystem.new('incr' => '{% increment %}')
|
||||||
assert_template_result '010', '{% increment %}{% increment %}{% render "incr" %}'
|
assert_template_result('010', '{% increment %}{% increment %}{% render "incr" %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_decrement_is_isolated_between_renders
|
def test_decrement_is_isolated_between_renders
|
||||||
Liquid::Template.file_system = StubFileSystem.new('decr' => '{% decrement %}')
|
Liquid::Template.file_system = StubFileSystem.new('decr' => '{% decrement %}')
|
||||||
assert_template_result '-1-2-1', '{% decrement %}{% decrement %}{% render "decr" %}'
|
assert_template_result('-1-2-1', '{% decrement %}{% decrement %}{% render "decr" %}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_includes_will_not_render_inside_render_tag
|
def test_includes_will_not_render_inside_render_tag
|
||||||
@@ -153,7 +127,10 @@ class RenderTagTest < Minitest::Test
|
|||||||
'test_include' => '{% include "foo" %}'
|
'test_include' => '{% include "foo" %}'
|
||||||
)
|
)
|
||||||
|
|
||||||
assert_template_result 'include usage is not allowed in this context', '{% render "test_include" %}'
|
exc = assert_raises(Liquid::DisabledError) do
|
||||||
|
Liquid::Template.parse('{% render "test_include" %}').render!
|
||||||
|
end
|
||||||
|
assert_equal('Liquid error: include usage is not allowed in this context', exc.message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_includes_will_not_render_inside_nested_sibling_tags
|
def test_includes_will_not_render_inside_nested_sibling_tags
|
||||||
@@ -163,6 +140,74 @@ class RenderTagTest < Minitest::Test
|
|||||||
'test_include' => '{% include "foo" %}'
|
'test_include' => '{% include "foo" %}'
|
||||||
)
|
)
|
||||||
|
|
||||||
assert_template_result 'include usage is not allowed in this contextinclude usage is not allowed in this context', '{% render "nested_render_with_sibling_include" %}'
|
output = Liquid::Template.parse('{% render "nested_render_with_sibling_include" %}').render
|
||||||
|
assert_equal('Liquid error: include usage is not allowed in this contextLiquid error: include usage is not allowed in this context', output)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_tag_with
|
||||||
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
|
'product' => "Product: {{ product.title }} ",
|
||||||
|
'product_alias' => "Product: {{ product.title }} ",
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_template_result("Product: Draft 151cm ",
|
||||||
|
"{% render 'product' with products[0] %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_tag_with_alias
|
||||||
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
|
'product' => "Product: {{ product.title }} ",
|
||||||
|
'product_alias' => "Product: {{ product.title }} ",
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_template_result("Product: Draft 151cm ",
|
||||||
|
"{% render 'product_alias' with products[0] as product %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_tag_for_alias
|
||||||
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
|
'product' => "Product: {{ product.title }} ",
|
||||||
|
'product_alias' => "Product: {{ product.title }} ",
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_template_result("Product: Draft 151cm Product: Element 155cm ",
|
||||||
|
"{% render 'product_alias' for products as product %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_tag_for
|
||||||
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
|
'product' => "Product: {{ product.title }} ",
|
||||||
|
'product_alias' => "Product: {{ product.title }} ",
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_template_result("Product: Draft 151cm Product: Element 155cm ",
|
||||||
|
"{% render 'product' for products %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_tag_forloop
|
||||||
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
|
'product' => "Product: {{ product.title }} {% if forloop.first %}first{% endif %} {% if forloop.last %}last{% endif %} index:{{ forloop.index }} ",
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_template_result("Product: Draft 151cm first index:1 Product: Element 155cm last index:2 ",
|
||||||
|
"{% render 'product' for products %}", "products" => [{ 'title' => 'Draft 151cm' }, { 'title' => 'Element 155cm' }])
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_tag_for_drop
|
||||||
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
|
'loop' => "{{ value.foo }}",
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_template_result("123",
|
||||||
|
"{% render 'loop' for loop as value %}", "loop" => TestEnumerable.new)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_render_tag_with_drop
|
||||||
|
Liquid::Template.file_system = StubFileSystem.new(
|
||||||
|
'loop' => "{{ value }}",
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_template_result("TestEnumerable",
|
||||||
|
"{% render 'loop' with loop as value %}", "loop" => TestEnumerable.new)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -176,11 +176,11 @@ class StandardTagTest < Minitest::Test
|
|||||||
# Example from the shopify forums
|
# Example from the shopify forums
|
||||||
code = "{% case collection.handle %}{% when 'menswear-jackets' %}{% assign ptitle = 'menswear' %}{% when 'menswear-t-shirts' %}{% assign ptitle = 'menswear' %}{% else %}{% assign ptitle = 'womenswear' %}{% endcase %}{{ ptitle }}"
|
code = "{% case collection.handle %}{% when 'menswear-jackets' %}{% assign ptitle = 'menswear' %}{% when 'menswear-t-shirts' %}{% assign ptitle = 'menswear' %}{% else %}{% assign ptitle = 'womenswear' %}{% endcase %}{{ ptitle }}"
|
||||||
template = Liquid::Template.parse(code)
|
template = Liquid::Template.parse(code)
|
||||||
assert_equal "menswear", template.render!("collection" => { 'handle' => 'menswear-jackets' })
|
assert_equal("menswear", template.render!("collection" => { 'handle' => 'menswear-jackets' }))
|
||||||
assert_equal "menswear", template.render!("collection" => { 'handle' => 'menswear-t-shirts' })
|
assert_equal("menswear", template.render!("collection" => { 'handle' => 'menswear-t-shirts' }))
|
||||||
assert_equal "womenswear", template.render!("collection" => { 'handle' => 'x' })
|
assert_equal("womenswear", template.render!("collection" => { 'handle' => 'x' }))
|
||||||
assert_equal "womenswear", template.render!("collection" => { 'handle' => 'y' })
|
assert_equal("womenswear", template.render!("collection" => { 'handle' => 'y' }))
|
||||||
assert_equal "womenswear", template.render!("collection" => { 'handle' => 'z' })
|
assert_equal("womenswear", template.render!("collection" => { 'handle' => 'z' }))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_case_when_or
|
def test_case_when_or
|
||||||
@@ -213,8 +213,13 @@ class StandardTagTest < Minitest::Test
|
|||||||
assert_template_result('', code, 'condition' => 'something else')
|
assert_template_result('', code, 'condition' => 'something else')
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_case_when_comma_and_blank_body
|
||||||
|
code = '{% case condition %}{% when 1, 2 %} {% assign r = "result" %} {% endcase %}{{ r }}'
|
||||||
|
assert_template_result('result', code, 'condition' => 2)
|
||||||
|
end
|
||||||
|
|
||||||
def test_assign
|
def test_assign
|
||||||
assert_template_result 'variable', '{% assign a = "variable"%}{{a}}'
|
assert_template_result('variable', '{% assign a = "variable"%}{{a}}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_assign_unassigned
|
def test_assign_unassigned
|
||||||
@@ -223,11 +228,11 @@ class StandardTagTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_assign_an_empty_string
|
def test_assign_an_empty_string
|
||||||
assert_template_result '', '{% assign a = ""%}{{a}}'
|
assert_template_result('', '{% assign a = ""%}{{a}}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_assign_is_global
|
def test_assign_is_global
|
||||||
assert_template_result 'variable', '{%for i in (1..2) %}{% assign a = "variable"%}{% endfor %}{{a}}'
|
assert_template_result('variable', '{%for i in (1..2) %}{% assign a = "variable"%}{% endfor %}{{a}}')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_case_detects_bad_syntax
|
def test_case_detects_bad_syntax
|
||||||
@@ -293,6 +298,6 @@ class StandardTagTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_multiline_tag
|
def test_multiline_tag
|
||||||
assert_template_result '0 1 2 3', "0{%\nfor i in (1..3)\n%} {{\ni\n}}{%\nendfor\n%}"
|
assert_template_result('0 1 2 3', "0{%\nfor i in (1..3)\n%} {{\ni\n}}{%\nendfor\n%}")
|
||||||
end
|
end
|
||||||
end # StandardTagTest
|
end # StandardTagTest
|
||||||
|
|||||||
@@ -7,75 +7,75 @@ class StatementsTest < Minitest::Test
|
|||||||
|
|
||||||
def test_true_eql_true
|
def test_true_eql_true
|
||||||
text = ' {% if true == true %} true {% else %} false {% endif %} '
|
text = ' {% if true == true %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text
|
assert_template_result(' true ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_true_not_eql_true
|
def test_true_not_eql_true
|
||||||
text = ' {% if true != true %} true {% else %} false {% endif %} '
|
text = ' {% if true != true %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' false ', text
|
assert_template_result(' false ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_true_lq_true
|
def test_true_lq_true
|
||||||
text = ' {% if 0 > 0 %} true {% else %} false {% endif %} '
|
text = ' {% if 0 > 0 %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' false ', text
|
assert_template_result(' false ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_one_lq_zero
|
def test_one_lq_zero
|
||||||
text = ' {% if 1 > 0 %} true {% else %} false {% endif %} '
|
text = ' {% if 1 > 0 %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text
|
assert_template_result(' true ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_zero_lq_one
|
def test_zero_lq_one
|
||||||
text = ' {% if 0 < 1 %} true {% else %} false {% endif %} '
|
text = ' {% if 0 < 1 %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text
|
assert_template_result(' true ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_zero_lq_or_equal_one
|
def test_zero_lq_or_equal_one
|
||||||
text = ' {% if 0 <= 0 %} true {% else %} false {% endif %} '
|
text = ' {% if 0 <= 0 %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text
|
assert_template_result(' true ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_zero_lq_or_equal_one_involving_nil
|
def test_zero_lq_or_equal_one_involving_nil
|
||||||
text = ' {% if null <= 0 %} true {% else %} false {% endif %} '
|
text = ' {% if null <= 0 %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' false ', text
|
assert_template_result(' false ', text)
|
||||||
|
|
||||||
text = ' {% if 0 <= null %} true {% else %} false {% endif %} '
|
text = ' {% if 0 <= null %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' false ', text
|
assert_template_result(' false ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_zero_lqq_or_equal_one
|
def test_zero_lqq_or_equal_one
|
||||||
text = ' {% if 0 >= 0 %} true {% else %} false {% endif %} '
|
text = ' {% if 0 >= 0 %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text
|
assert_template_result(' true ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strings
|
def test_strings
|
||||||
text = " {% if 'test' == 'test' %} true {% else %} false {% endif %} "
|
text = " {% if 'test' == 'test' %} true {% else %} false {% endif %} "
|
||||||
assert_template_result ' true ', text
|
assert_template_result(' true ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_strings_not_equal
|
def test_strings_not_equal
|
||||||
text = " {% if 'test' != 'test' %} true {% else %} false {% endif %} "
|
text = " {% if 'test' != 'test' %} true {% else %} false {% endif %} "
|
||||||
assert_template_result ' false ', text
|
assert_template_result(' false ', text)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_var_strings_equal
|
def test_var_strings_equal
|
||||||
text = ' {% if var == "hello there!" %} true {% else %} false {% endif %} '
|
text = ' {% if var == "hello there!" %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text, 'var' => 'hello there!'
|
assert_template_result(' true ', text, 'var' => 'hello there!')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_var_strings_are_not_equal
|
def test_var_strings_are_not_equal
|
||||||
text = ' {% if "hello there!" == var %} true {% else %} false {% endif %} '
|
text = ' {% if "hello there!" == var %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text, 'var' => 'hello there!'
|
assert_template_result(' true ', text, 'var' => 'hello there!')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_var_and_long_string_are_equal
|
def test_var_and_long_string_are_equal
|
||||||
text = " {% if var == 'hello there!' %} true {% else %} false {% endif %} "
|
text = " {% if var == 'hello there!' %} true {% else %} false {% endif %} "
|
||||||
assert_template_result ' true ', text, 'var' => 'hello there!'
|
assert_template_result(' true ', text, 'var' => 'hello there!')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_var_and_long_string_are_equal_backwards
|
def test_var_and_long_string_are_equal_backwards
|
||||||
text = " {% if 'hello there!' == var %} true {% else %} false {% endif %} "
|
text = " {% if 'hello there!' == var %} true {% else %} false {% endif %} "
|
||||||
assert_template_result ' true ', text, 'var' => 'hello there!'
|
assert_template_result(' true ', text, 'var' => 'hello there!')
|
||||||
end
|
end
|
||||||
|
|
||||||
# def test_is_nil
|
# def test_is_nil
|
||||||
@@ -87,27 +87,27 @@ class StatementsTest < Minitest::Test
|
|||||||
|
|
||||||
def test_is_collection_empty
|
def test_is_collection_empty
|
||||||
text = ' {% if array == empty %} true {% else %} false {% endif %} '
|
text = ' {% if array == empty %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text, 'array' => []
|
assert_template_result(' true ', text, 'array' => [])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_is_not_collection_empty
|
def test_is_not_collection_empty
|
||||||
text = ' {% if array == empty %} true {% else %} false {% endif %} '
|
text = ' {% if array == empty %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' false ', text, 'array' => [1, 2, 3]
|
assert_template_result(' false ', text, 'array' => [1, 2, 3])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nil
|
def test_nil
|
||||||
text = ' {% if var == nil %} true {% else %} false {% endif %} '
|
text = ' {% if var == nil %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text, 'var' => nil
|
assert_template_result(' true ', text, 'var' => nil)
|
||||||
|
|
||||||
text = ' {% if var == null %} true {% else %} false {% endif %} '
|
text = ' {% if var == null %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text, 'var' => nil
|
assert_template_result(' true ', text, 'var' => nil)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_not_nil
|
def test_not_nil
|
||||||
text = ' {% if var != nil %} true {% else %} false {% endif %} '
|
text = ' {% if var != nil %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text, 'var' => 1
|
assert_template_result(' true ', text, 'var' => 1)
|
||||||
|
|
||||||
text = ' {% if var != null %} true {% else %} false {% endif %} '
|
text = ' {% if var != null %} true {% else %} false {% endif %} '
|
||||||
assert_template_result ' true ', text, 'var' => 1
|
assert_template_result(' true ', text, 'var' => 1)
|
||||||
end
|
end
|
||||||
end # StatementsTest
|
end # StatementsTest
|
||||||
|
|||||||
@@ -19,10 +19,10 @@ class UnlessElseTagTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_unless_in_loop
|
def test_unless_in_loop
|
||||||
assert_template_result '23', '{% for i in choices %}{% unless i %}{{ forloop.index }}{% endunless %}{% endfor %}', 'choices' => [1, nil, false]
|
assert_template_result('23', '{% for i in choices %}{% unless i %}{{ forloop.index }}{% endunless %}{% endfor %}', 'choices' => [1, nil, false])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_unless_else_in_loop
|
def test_unless_else_in_loop
|
||||||
assert_template_result ' TRUE 2 3 ', '{% for i in choices %}{% unless i %} {{ forloop.index }} {% else %} TRUE {% endunless %}{% endfor %}', 'choices' => [1, nil, false]
|
assert_template_result(' TRUE 2 3 ', '{% for i in choices %}{% unless i %} {{ forloop.index }} {% else %} TRUE {% endunless %}{% endfor %}', 'choices' => [1, nil, false])
|
||||||
end
|
end
|
||||||
end # UnlessElseTest
|
end # UnlessElseTest
|
||||||
|
|||||||
@@ -40,8 +40,8 @@ class TemplateTest < Minitest::Test
|
|||||||
|
|
||||||
def test_instance_assigns_persist_on_same_template_object_between_parses
|
def test_instance_assigns_persist_on_same_template_object_between_parses
|
||||||
t = Template.new
|
t = Template.new
|
||||||
assert_equal 'from instance assigns', t.parse("{% assign foo = 'from instance assigns' %}{{ foo }}").render!
|
assert_equal('from instance assigns', t.parse("{% assign foo = 'from instance assigns' %}{{ foo }}").render!)
|
||||||
assert_equal 'from instance assigns', t.parse("{{ foo }}").render!
|
assert_equal('from instance assigns', t.parse("{{ foo }}").render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_warnings_is_not_exponential_time
|
def test_warnings_is_not_exponential_time
|
||||||
@@ -51,32 +51,32 @@ class TemplateTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
t = Template.parse(str)
|
t = Template.parse(str)
|
||||||
assert_equal [], Timeout.timeout(1) { t.warnings }
|
assert_equal([], Timeout.timeout(1) { t.warnings })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_instance_assigns_persist_on_same_template_parsing_between_renders
|
def test_instance_assigns_persist_on_same_template_parsing_between_renders
|
||||||
t = Template.new.parse("{{ foo }}{% assign foo = 'foo' %}{{ foo }}")
|
t = Template.new.parse("{{ foo }}{% assign foo = 'foo' %}{{ foo }}")
|
||||||
assert_equal 'foo', t.render!
|
assert_equal('foo', t.render!)
|
||||||
assert_equal 'foofoo', t.render!
|
assert_equal('foofoo', t.render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_custom_assigns_do_not_persist_on_same_template
|
def test_custom_assigns_do_not_persist_on_same_template
|
||||||
t = Template.new
|
t = Template.new
|
||||||
assert_equal 'from custom assigns', t.parse("{{ foo }}").render!('foo' => 'from custom assigns')
|
assert_equal('from custom assigns', t.parse("{{ foo }}").render!('foo' => 'from custom assigns'))
|
||||||
assert_equal '', t.parse("{{ foo }}").render!
|
assert_equal('', t.parse("{{ foo }}").render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_custom_assigns_squash_instance_assigns
|
def test_custom_assigns_squash_instance_assigns
|
||||||
t = Template.new
|
t = Template.new
|
||||||
assert_equal 'from instance assigns', t.parse("{% assign foo = 'from instance assigns' %}{{ foo }}").render!
|
assert_equal('from instance assigns', t.parse("{% assign foo = 'from instance assigns' %}{{ foo }}").render!)
|
||||||
assert_equal 'from custom assigns', t.parse("{{ foo }}").render!('foo' => 'from custom assigns')
|
assert_equal('from custom assigns', t.parse("{{ foo }}").render!('foo' => 'from custom assigns'))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_persistent_assigns_squash_instance_assigns
|
def test_persistent_assigns_squash_instance_assigns
|
||||||
t = Template.new
|
t = Template.new
|
||||||
assert_equal 'from instance assigns', t.parse("{% assign foo = 'from instance assigns' %}{{ foo }}").render!
|
assert_equal('from instance assigns', t.parse("{% assign foo = 'from instance assigns' %}{{ foo }}").render!)
|
||||||
t.assigns['foo'] = 'from persistent assigns'
|
t.assigns['foo'] = 'from persistent assigns'
|
||||||
assert_equal 'from persistent assigns', t.parse("{{ foo }}").render!
|
assert_equal('from persistent assigns', t.parse("{{ foo }}").render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_lambda_is_called_once_from_persistent_assigns_over_multiple_parses_and_renders
|
def test_lambda_is_called_once_from_persistent_assigns_over_multiple_parses_and_renders
|
||||||
@@ -85,9 +85,9 @@ class TemplateTest < Minitest::Test
|
|||||||
@global ||= 0
|
@global ||= 0
|
||||||
@global += 1
|
@global += 1
|
||||||
}
|
}
|
||||||
assert_equal '1', t.parse("{{number}}").render!
|
assert_equal('1', t.parse("{{number}}").render!)
|
||||||
assert_equal '1', t.parse("{{number}}").render!
|
assert_equal('1', t.parse("{{number}}").render!)
|
||||||
assert_equal '1', t.render!
|
assert_equal('1', t.render!)
|
||||||
@global = nil
|
@global = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -97,148 +97,111 @@ class TemplateTest < Minitest::Test
|
|||||||
@global ||= 0
|
@global ||= 0
|
||||||
@global += 1
|
@global += 1
|
||||||
} }
|
} }
|
||||||
assert_equal '1', t.parse("{{number}}").render!(assigns)
|
assert_equal('1', t.parse("{{number}}").render!(assigns))
|
||||||
assert_equal '1', t.parse("{{number}}").render!(assigns)
|
assert_equal('1', t.parse("{{number}}").render!(assigns))
|
||||||
assert_equal '1', t.render!(assigns)
|
assert_equal('1', t.render!(assigns))
|
||||||
@global = nil
|
@global = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_resource_limits_works_with_custom_length_method
|
def test_resource_limits_works_with_custom_length_method
|
||||||
t = Template.parse("{% assign foo = bar %}")
|
t = Template.parse("{% assign foo = bar %}")
|
||||||
t.resource_limits.render_length_limit = 42
|
t.resource_limits.render_length_limit = 42
|
||||||
assert_equal "", t.render!("bar" => SomethingWithLength.new)
|
assert_equal("", t.render!("bar" => SomethingWithLength.new))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_resource_limits_render_length
|
def test_resource_limits_render_length
|
||||||
t = Template.parse("0123456789")
|
t = Template.parse("0123456789")
|
||||||
t.resource_limits.render_length_limit = 5
|
t.resource_limits.render_length_limit = 9
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
assert t.resource_limits.reached?
|
assert(t.resource_limits.reached?)
|
||||||
|
|
||||||
t.resource_limits.render_length_limit = 10
|
t.resource_limits.render_length_limit = 10
|
||||||
assert_equal "0123456789", t.render!
|
assert_equal("0123456789", t.render!)
|
||||||
refute_nil t.resource_limits.render_length
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_resource_limits_render_score
|
def test_resource_limits_render_score
|
||||||
t = Template.parse("{% for a in (1..10) %} {% for a in (1..10) %} foo {% endfor %} {% endfor %}")
|
t = Template.parse("{% for a in (1..10) %} {% for a in (1..10) %} foo {% endfor %} {% endfor %}")
|
||||||
t.resource_limits.render_score_limit = 50
|
t.resource_limits.render_score_limit = 50
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
assert t.resource_limits.reached?
|
assert(t.resource_limits.reached?)
|
||||||
|
|
||||||
t = Template.parse("{% for a in (1..100) %} foo {% endfor %}")
|
t = Template.parse("{% for a in (1..100) %} foo {% endfor %}")
|
||||||
t.resource_limits.render_score_limit = 50
|
t.resource_limits.render_score_limit = 50
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
assert t.resource_limits.reached?
|
assert(t.resource_limits.reached?)
|
||||||
|
|
||||||
t.resource_limits.render_score_limit = 200
|
t.resource_limits.render_score_limit = 200
|
||||||
assert_equal (" foo " * 100), t.render!
|
assert_equal((" foo " * 100), t.render!)
|
||||||
refute_nil t.resource_limits.render_score
|
refute_nil(t.resource_limits.render_score)
|
||||||
end
|
|
||||||
|
|
||||||
def test_resource_limits_assign_score
|
|
||||||
t = Template.parse("{% assign foo = 42 %}{% assign bar = 23 %}")
|
|
||||||
t.resource_limits.assign_score_limit = 1
|
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
|
||||||
assert t.resource_limits.reached?
|
|
||||||
|
|
||||||
t.resource_limits.assign_score_limit = 2
|
|
||||||
assert_equal "", t.render!
|
|
||||||
refute_nil t.resource_limits.assign_score
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_resource_limits_assign_score_counts_bytes_not_characters
|
|
||||||
t = Template.parse("{% assign foo = 'すごい' %}")
|
|
||||||
t.render
|
|
||||||
assert_equal 9, t.resource_limits.assign_score
|
|
||||||
|
|
||||||
t = Template.parse("{% capture foo %}すごい{% endcapture %}")
|
|
||||||
t.render
|
|
||||||
assert_equal 9, t.resource_limits.assign_score
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_resource_limits_assign_score_nested
|
|
||||||
t = Template.parse("{% assign foo = 'aaaa' | reverse %}")
|
|
||||||
|
|
||||||
t.resource_limits.assign_score_limit = 3
|
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
|
||||||
assert t.resource_limits.reached?
|
|
||||||
|
|
||||||
t.resource_limits.assign_score_limit = 5
|
|
||||||
assert_equal "", t.render!
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_resource_limits_aborts_rendering_after_first_error
|
def test_resource_limits_aborts_rendering_after_first_error
|
||||||
t = Template.parse("{% for a in (1..100) %} foo1 {% endfor %} bar {% for a in (1..100) %} foo2 {% endfor %}")
|
t = Template.parse("{% for a in (1..100) %} foo1 {% endfor %} bar {% for a in (1..100) %} foo2 {% endfor %}")
|
||||||
t.resource_limits.render_score_limit = 50
|
t.resource_limits.render_score_limit = 50
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
assert t.resource_limits.reached?
|
assert(t.resource_limits.reached?)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_resource_limits_hash_in_template_gets_updated_even_if_no_limits_are_set
|
def test_resource_limits_hash_in_template_gets_updated_even_if_no_limits_are_set
|
||||||
t = Template.parse("{% for a in (1..100) %} {% assign foo = 1 %} {% endfor %}")
|
t = Template.parse("{% for a in (1..100) %}x{% assign foo = 1 %} {% endfor %}")
|
||||||
t.render!
|
t.render!
|
||||||
assert t.resource_limits.assign_score > 0
|
assert(t.resource_limits.assign_score > 0)
|
||||||
assert t.resource_limits.render_score > 0
|
assert(t.resource_limits.render_score > 0)
|
||||||
assert t.resource_limits.render_length > 0
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_length_persists_between_blocks
|
def test_render_length_persists_between_blocks
|
||||||
t = Template.parse("{% if true %}aaaa{% endif %}")
|
t = Template.parse("{% if true %}aaaa{% endif %}")
|
||||||
t.resource_limits.render_length_limit = 7
|
t.resource_limits.render_length_limit = 3
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
t.resource_limits.render_length_limit = 8
|
t.resource_limits.render_length_limit = 4
|
||||||
assert_equal "aaaa", t.render
|
assert_equal("aaaa", t.render)
|
||||||
|
|
||||||
t = Template.parse("{% if true %}aaaa{% endif %}{% if true %}bbb{% endif %}")
|
t = Template.parse("{% if true %}aaaa{% endif %}{% if true %}bbb{% endif %}")
|
||||||
t.resource_limits.render_length_limit = 13
|
t.resource_limits.render_length_limit = 6
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
t.resource_limits.render_length_limit = 14
|
t.resource_limits.render_length_limit = 7
|
||||||
assert_equal "aaaabbb", t.render
|
assert_equal("aaaabbb", t.render)
|
||||||
|
|
||||||
t = Template.parse("{% if true %}a{% endif %}{% if true %}b{% endif %}{% if true %}a{% endif %}{% if true %}b{% endif %}{% if true %}a{% endif %}{% if true %}b{% endif %}")
|
t = Template.parse("{% if true %}a{% endif %}{% if true %}b{% endif %}{% if true %}a{% endif %}{% if true %}b{% endif %}{% if true %}a{% endif %}{% if true %}b{% endif %}")
|
||||||
t.resource_limits.render_length_limit = 5
|
t.resource_limits.render_length_limit = 5
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
t.resource_limits.render_length_limit = 11
|
t.resource_limits.render_length_limit = 6
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("ababab", t.render)
|
||||||
t.resource_limits.render_length_limit = 12
|
|
||||||
assert_equal "ababab", t.render
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_length_uses_number_of_bytes_not_characters
|
def test_render_length_uses_number_of_bytes_not_characters
|
||||||
t = Template.parse("{% if true %}すごい{% endif %}")
|
t = Template.parse("{% if true %}すごい{% endif %}")
|
||||||
t.resource_limits.render_length_limit = 10
|
t.resource_limits.render_length_limit = 8
|
||||||
assert_equal "Liquid error: Memory limits exceeded", t.render
|
assert_equal("Liquid error: Memory limits exceeded", t.render)
|
||||||
t.resource_limits.render_length_limit = 18
|
t.resource_limits.render_length_limit = 9
|
||||||
assert_equal "すごい", t.render
|
assert_equal("すごい", t.render)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_default_resource_limits_unaffected_by_render_with_context
|
def test_default_resource_limits_unaffected_by_render_with_context
|
||||||
context = Context.new
|
context = Context.new
|
||||||
t = Template.parse("{% for a in (1..100) %} {% assign foo = 1 %} {% endfor %}")
|
t = Template.parse("{% for a in (1..100) %}x{% assign foo = 1 %} {% endfor %}")
|
||||||
t.render!(context)
|
t.render!(context)
|
||||||
assert context.resource_limits.assign_score > 0
|
assert(context.resource_limits.assign_score > 0)
|
||||||
assert context.resource_limits.render_score > 0
|
assert(context.resource_limits.render_score > 0)
|
||||||
assert context.resource_limits.render_length > 0
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_can_use_drop_as_context
|
def test_can_use_drop_as_context
|
||||||
t = Template.new
|
t = Template.new
|
||||||
t.registers['lulz'] = 'haha'
|
t.registers['lulz'] = 'haha'
|
||||||
drop = TemplateContextDrop.new
|
drop = TemplateContextDrop.new
|
||||||
assert_equal 'fizzbuzz', t.parse('{{foo}}').render!(drop)
|
assert_equal('fizzbuzz', t.parse('{{foo}}').render!(drop))
|
||||||
assert_equal 'bar', t.parse('{{bar}}').render!(drop)
|
assert_equal('bar', t.parse('{{bar}}').render!(drop))
|
||||||
assert_equal 'haha', t.parse("{{baz}}").render!(drop)
|
assert_equal('haha', t.parse("{{baz}}").render!(drop))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_bang_force_rethrow_errors_on_passed_context
|
def test_render_bang_force_rethrow_errors_on_passed_context
|
||||||
context = Context.new('drop' => ErroneousDrop.new)
|
context = Context.new('drop' => ErroneousDrop.new)
|
||||||
t = Template.new.parse('{{ drop.bad_method }}')
|
t = Template.new.parse('{{ drop.bad_method }}')
|
||||||
|
|
||||||
e = assert_raises RuntimeError do
|
e = assert_raises(RuntimeError) do
|
||||||
t.render!(context)
|
t.render!(context)
|
||||||
end
|
end
|
||||||
assert_equal 'ruby error in drop', e.message
|
assert_equal('ruby error in drop', e.message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_exception_renderer_that_returns_string
|
def test_exception_renderer_that_returns_string
|
||||||
@@ -250,8 +213,8 @@ class TemplateTest < Minitest::Test
|
|||||||
|
|
||||||
output = Template.parse("{{ 1 | divided_by: 0 }}").render({}, exception_renderer: handler)
|
output = Template.parse("{{ 1 | divided_by: 0 }}").render({}, exception_renderer: handler)
|
||||||
|
|
||||||
assert exception.is_a?(Liquid::ZeroDivisionError)
|
assert(exception.is_a?(Liquid::ZeroDivisionError))
|
||||||
assert_equal '<!-- error -->', output
|
assert_equal('<!-- error -->', output)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_exception_renderer_that_raises
|
def test_exception_renderer_that_raises
|
||||||
@@ -262,50 +225,49 @@ class TemplateTest < Minitest::Test
|
|||||||
raise
|
raise
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
assert exception.is_a?(Liquid::ZeroDivisionError)
|
assert(exception.is_a?(Liquid::ZeroDivisionError))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_global_filter_option_on_render
|
def test_global_filter_option_on_render
|
||||||
global_filter_proc = ->(output) { "#{output} filtered" }
|
global_filter_proc = ->(output) { "#{output} filtered" }
|
||||||
rendered_template = Template.parse("{{name}}").render({ "name" => "bob" }, global_filter: global_filter_proc)
|
rendered_template = Template.parse("{{name}}").render({ "name" => "bob" }, global_filter: global_filter_proc)
|
||||||
|
|
||||||
assert_equal 'bob filtered', rendered_template
|
assert_equal('bob filtered', rendered_template)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_global_filter_option_when_native_filters_exist
|
def test_global_filter_option_when_native_filters_exist
|
||||||
global_filter_proc = ->(output) { "#{output} filtered" }
|
global_filter_proc = ->(output) { "#{output} filtered" }
|
||||||
rendered_template = Template.parse("{{name | upcase}}").render({ "name" => "bob" }, global_filter: global_filter_proc)
|
rendered_template = Template.parse("{{name | upcase}}").render({ "name" => "bob" }, global_filter: global_filter_proc)
|
||||||
|
|
||||||
assert_equal 'BOB filtered', rendered_template
|
assert_equal('BOB filtered', rendered_template)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_undefined_variables
|
def test_undefined_variables
|
||||||
t = Template.parse("{{x}} {{y}} {{z.a}} {{z.b}} {{z.c.d}}")
|
t = Template.parse("{{x}} {{y}} {{z.a}} {{z.b}} {{z.c.d}}")
|
||||||
result = t.render({ 'x' => 33, 'z' => { 'a' => 32, 'c' => { 'e' => 31 } } }, strict_variables: true)
|
result = t.render({ 'x' => 33, 'z' => { 'a' => 32, 'c' => { 'e' => 31 } } }, strict_variables: true)
|
||||||
|
|
||||||
assert_equal '33 32 ', result
|
assert_equal('33 32 ', result)
|
||||||
assert_equal 3, t.errors.count
|
assert_equal(3, t.errors.count)
|
||||||
assert_instance_of Liquid::UndefinedVariable, t.errors[0]
|
assert_instance_of(Liquid::UndefinedVariable, t.errors[0])
|
||||||
assert_equal 'Liquid error: undefined variable y', t.errors[0].message
|
assert_equal('Liquid error: undefined variable y', t.errors[0].message)
|
||||||
assert_instance_of Liquid::UndefinedVariable, t.errors[1]
|
assert_instance_of(Liquid::UndefinedVariable, t.errors[1])
|
||||||
assert_equal 'Liquid error: undefined variable b', t.errors[1].message
|
assert_equal('Liquid error: undefined variable b', t.errors[1].message)
|
||||||
assert_instance_of Liquid::UndefinedVariable, t.errors[2]
|
assert_instance_of(Liquid::UndefinedVariable, t.errors[2])
|
||||||
assert_equal 'Liquid error: undefined variable d', t.errors[2].message
|
assert_equal('Liquid error: undefined variable d', t.errors[2].message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nil_value_does_not_raise
|
def test_nil_value_does_not_raise
|
||||||
Liquid::Template.error_mode = :strict
|
t = Template.parse("some{{x}}thing", error_mode: :strict)
|
||||||
t = Template.parse("some{{x}}thing")
|
|
||||||
result = t.render!({ 'x' => nil }, strict_variables: true)
|
result = t.render!({ 'x' => nil }, strict_variables: true)
|
||||||
|
|
||||||
assert_equal 0, t.errors.count
|
assert_equal(0, t.errors.count)
|
||||||
assert_equal 'something', result
|
assert_equal('something', result)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_undefined_variables_raise
|
def test_undefined_variables_raise
|
||||||
t = Template.parse("{{x}} {{y}} {{z.a}} {{z.b}} {{z.c.d}}")
|
t = Template.parse("{{x}} {{y}} {{z.a}} {{z.b}} {{z.c.d}}")
|
||||||
|
|
||||||
assert_raises UndefinedVariable do
|
assert_raises(UndefinedVariable) do
|
||||||
t.render!({ 'x' => 33, 'z' => { 'a' => 32, 'c' => { 'e' => 31 } } }, strict_variables: true)
|
t.render!({ 'x' => 33, 'z' => { 'a' => 32, 'c' => { 'e' => 31 } } }, strict_variables: true)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -315,16 +277,16 @@ class TemplateTest < Minitest::Test
|
|||||||
t = Template.new.parse('{{ foo }} {{ woot }}')
|
t = Template.new.parse('{{ foo }} {{ woot }}')
|
||||||
result = t.render(d, strict_variables: true)
|
result = t.render(d, strict_variables: true)
|
||||||
|
|
||||||
assert_equal 'foo ', result
|
assert_equal('foo ', result)
|
||||||
assert_equal 1, t.errors.count
|
assert_equal(1, t.errors.count)
|
||||||
assert_instance_of Liquid::UndefinedDropMethod, t.errors[0]
|
assert_instance_of(Liquid::UndefinedDropMethod, t.errors[0])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_undefined_drop_methods_raise
|
def test_undefined_drop_methods_raise
|
||||||
d = DropWithUndefinedMethod.new
|
d = DropWithUndefinedMethod.new
|
||||||
t = Template.new.parse('{{ foo }} {{ woot }}')
|
t = Template.new.parse('{{ foo }} {{ woot }}')
|
||||||
|
|
||||||
assert_raises UndefinedDropMethod do
|
assert_raises(UndefinedDropMethod) do
|
||||||
t.render!(d, strict_variables: true)
|
t.render!(d, strict_variables: true)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -338,16 +300,16 @@ class TemplateTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
result = t.render({ 'a' => 123, 'x' => 'foo' }, filters: [filters], strict_filters: true)
|
result = t.render({ 'a' => 123, 'x' => 'foo' }, filters: [filters], strict_filters: true)
|
||||||
|
|
||||||
assert_equal '123 ', result
|
assert_equal('123 ', result)
|
||||||
assert_equal 1, t.errors.count
|
assert_equal(1, t.errors.count)
|
||||||
assert_instance_of Liquid::UndefinedFilter, t.errors[0]
|
assert_instance_of(Liquid::UndefinedFilter, t.errors[0])
|
||||||
assert_equal 'Liquid error: undefined filter somefilter1', t.errors[0].message
|
assert_equal('Liquid error: undefined filter somefilter1', t.errors[0].message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_undefined_filters_raise
|
def test_undefined_filters_raise
|
||||||
t = Template.parse("{{x | somefilter1 | upcase | somefilter2}}")
|
t = Template.parse("{{x | somefilter1 | upcase | somefilter2}}")
|
||||||
|
|
||||||
assert_raises UndefinedFilter do
|
assert_raises(UndefinedFilter) do
|
||||||
t.render!({ 'x' => 'foo' }, strict_filters: true)
|
t.render!({ 'x' => 'foo' }, strict_filters: true)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -355,10 +317,10 @@ class TemplateTest < Minitest::Test
|
|||||||
def test_using_range_literal_works_as_expected
|
def test_using_range_literal_works_as_expected
|
||||||
t = Template.parse("{% assign foo = (x..y) %}{{ foo }}")
|
t = Template.parse("{% assign foo = (x..y) %}{{ foo }}")
|
||||||
result = t.render('x' => 1, 'y' => 5)
|
result = t.render('x' => 1, 'y' => 5)
|
||||||
assert_equal '1..5', result
|
assert_equal('1..5', result)
|
||||||
|
|
||||||
t = Template.parse("{% assign nums = (x..y) %}{% for num in nums %}{{ num }}{% endfor %}")
|
t = Template.parse("{% assign nums = (x..y) %}{% for num in nums %}{{ num }}{% endfor %}")
|
||||||
result = t.render('x' => 1, 'y' => 5)
|
result = t.render('x' => 1, 'y' => 5)
|
||||||
assert_equal '12345', result
|
assert_equal('12345', result)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -528,4 +528,36 @@ class TrimModeTest < Minitest::Test
|
|||||||
END_EXPECTED
|
END_EXPECTED
|
||||||
assert_template_result(expected, text)
|
assert_template_result(expected, text)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_pre_trim_blank_preceding_text
|
||||||
|
template = Liquid::Template.parse("\n{%- raw %}{% endraw %}")
|
||||||
|
assert_equal("", template.render)
|
||||||
|
|
||||||
|
template = Liquid::Template.parse("\n{%- if true %}{% endif %}")
|
||||||
|
assert_equal("", template.render)
|
||||||
|
|
||||||
|
template = Liquid::Template.parse("{{ 'B' }} \n{%- if true %}C{% endif %}")
|
||||||
|
assert_equal("BC", template.render)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_bug_compatible_pre_trim
|
||||||
|
template = Liquid::Template.parse("\n {%- raw %}{% endraw %}", bug_compatible_whitespace_trimming: true)
|
||||||
|
assert_equal("\n", template.render)
|
||||||
|
|
||||||
|
template = Liquid::Template.parse("\n {%- if true %}{% endif %}", bug_compatible_whitespace_trimming: true)
|
||||||
|
assert_equal("\n", template.render)
|
||||||
|
|
||||||
|
template = Liquid::Template.parse("{{ 'B' }} \n{%- if true %}C{% endif %}", bug_compatible_whitespace_trimming: true)
|
||||||
|
assert_equal("B C", template.render)
|
||||||
|
|
||||||
|
template = Liquid::Template.parse("B\n {%- raw %}{% endraw %}", bug_compatible_whitespace_trimming: true)
|
||||||
|
assert_equal("B", template.render)
|
||||||
|
|
||||||
|
template = Liquid::Template.parse("B\n {%- if true %}{% endif %}", bug_compatible_whitespace_trimming: true)
|
||||||
|
assert_equal("B", template.render)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_trim_blank
|
||||||
|
assert_template_result('foobar', 'foo {{-}} bar')
|
||||||
|
end
|
||||||
end # TrimModeTest
|
end # TrimModeTest
|
||||||
|
|||||||
@@ -7,92 +7,101 @@ class VariableTest < Minitest::Test
|
|||||||
|
|
||||||
def test_simple_variable
|
def test_simple_variable
|
||||||
template = Template.parse(%({{test}}))
|
template = Template.parse(%({{test}}))
|
||||||
assert_equal 'worked', template.render!('test' => 'worked')
|
assert_equal('worked', template.render!('test' => 'worked'))
|
||||||
assert_equal 'worked wonderfully', template.render!('test' => 'worked wonderfully')
|
assert_equal('worked wonderfully', template.render!('test' => 'worked wonderfully'))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_render_calls_to_liquid
|
def test_variable_render_calls_to_liquid
|
||||||
assert_template_result 'foobar', '{{ foo }}', 'foo' => ThingWithToLiquid.new
|
assert_template_result('foobar', '{{ foo }}', 'foo' => ThingWithToLiquid.new)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_simple_with_whitespaces
|
def test_simple_with_whitespaces
|
||||||
template = Template.parse(%( {{ test }} ))
|
template = Template.parse(%( {{ test }} ))
|
||||||
assert_equal ' worked ', template.render!('test' => 'worked')
|
assert_equal(' worked ', template.render!('test' => 'worked'))
|
||||||
assert_equal ' worked wonderfully ', template.render!('test' => 'worked wonderfully')
|
assert_equal(' worked wonderfully ', template.render!('test' => 'worked wonderfully'))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_expression_with_whitespace_in_square_brackets
|
||||||
|
assert_template_result('result', "{{ a[ 'b' ] }}", 'a' => { 'b' => 'result' })
|
||||||
|
assert_template_result('result', "{{ a[ [ 'b' ] ] }}", 'b' => 'c', 'a' => { 'c' => 'result' })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_ignore_unknown
|
def test_ignore_unknown
|
||||||
template = Template.parse(%({{ test }}))
|
template = Template.parse(%({{ test }}))
|
||||||
assert_equal '', template.render!
|
assert_equal('', template.render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_using_blank_as_variable_name
|
def test_using_blank_as_variable_name
|
||||||
template = Template.parse("{% assign foo = blank %}{{ foo }}")
|
template = Template.parse("{% assign foo = blank %}{{ foo }}")
|
||||||
assert_equal '', template.render!
|
assert_equal('', template.render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_using_empty_as_variable_name
|
def test_using_empty_as_variable_name
|
||||||
template = Template.parse("{% assign foo = empty %}{{ foo }}")
|
template = Template.parse("{% assign foo = empty %}{{ foo }}")
|
||||||
assert_equal '', template.render!
|
assert_equal('', template.render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_hash_scoping
|
def test_hash_scoping
|
||||||
template = Template.parse(%({{ test.test }}))
|
assert_template_result('worked', "{{ test.test }}", 'test' => { 'test' => 'worked' })
|
||||||
assert_equal 'worked', template.render!('test' => { 'test' => 'worked' })
|
assert_template_result('worked', "{{ test . test }}", 'test' => { 'test' => 'worked' })
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_false_renders_as_false
|
def test_false_renders_as_false
|
||||||
assert_equal 'false', Template.parse("{{ foo }}").render!('foo' => false)
|
assert_equal('false', Template.parse("{{ foo }}").render!('foo' => false))
|
||||||
assert_equal 'false', Template.parse("{{ false }}").render!
|
assert_equal('false', Template.parse("{{ false }}").render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nil_renders_as_empty_string
|
def test_nil_renders_as_empty_string
|
||||||
assert_equal '', Template.parse("{{ nil }}").render!
|
assert_equal('', Template.parse("{{ nil }}").render!)
|
||||||
assert_equal 'cat', Template.parse("{{ nil | append: 'cat' }}").render!
|
assert_equal('cat', Template.parse("{{ nil | append: 'cat' }}").render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_preset_assigns
|
def test_preset_assigns
|
||||||
template = Template.parse(%({{ test }}))
|
template = Template.parse(%({{ test }}))
|
||||||
template.assigns['test'] = 'worked'
|
template.assigns['test'] = 'worked'
|
||||||
assert_equal 'worked', template.render!
|
assert_equal('worked', template.render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_reuse_parsed_template
|
def test_reuse_parsed_template
|
||||||
template = Template.parse(%({{ greeting }} {{ name }}))
|
template = Template.parse(%({{ greeting }} {{ name }}))
|
||||||
template.assigns['greeting'] = 'Goodbye'
|
template.assigns['greeting'] = 'Goodbye'
|
||||||
assert_equal 'Hello Tobi', template.render!('greeting' => 'Hello', 'name' => 'Tobi')
|
assert_equal('Hello Tobi', template.render!('greeting' => 'Hello', 'name' => 'Tobi'))
|
||||||
assert_equal 'Hello ', template.render!('greeting' => 'Hello', 'unknown' => 'Tobi')
|
assert_equal('Hello ', template.render!('greeting' => 'Hello', 'unknown' => 'Tobi'))
|
||||||
assert_equal 'Hello Brian', template.render!('greeting' => 'Hello', 'name' => 'Brian')
|
assert_equal('Hello Brian', template.render!('greeting' => 'Hello', 'name' => 'Brian'))
|
||||||
assert_equal 'Goodbye Brian', template.render!('name' => 'Brian')
|
assert_equal('Goodbye Brian', template.render!('name' => 'Brian'))
|
||||||
assert_equal({ 'greeting' => 'Goodbye' }, template.assigns)
|
assert_equal({ 'greeting' => 'Goodbye' }, template.assigns)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_assigns_not_polluted_from_template
|
def test_assigns_not_polluted_from_template
|
||||||
template = Template.parse(%({{ test }}{% assign test = 'bar' %}{{ test }}))
|
template = Template.parse(%({{ test }}{% assign test = 'bar' %}{{ test }}))
|
||||||
template.assigns['test'] = 'baz'
|
template.assigns['test'] = 'baz'
|
||||||
assert_equal 'bazbar', template.render!
|
assert_equal('bazbar', template.render!)
|
||||||
assert_equal 'bazbar', template.render!
|
assert_equal('bazbar', template.render!)
|
||||||
assert_equal 'foobar', template.render!('test' => 'foo')
|
assert_equal('foobar', template.render!('test' => 'foo'))
|
||||||
assert_equal 'bazbar', template.render!
|
assert_equal('bazbar', template.render!)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_hash_with_default_proc
|
def test_hash_with_default_proc
|
||||||
template = Template.parse(%(Hello {{ test }}))
|
template = Template.parse(%(Hello {{ test }}))
|
||||||
assigns = Hash.new { |_h, k| raise "Unknown variable '#{k}'" }
|
assigns = Hash.new { |_h, k| raise "Unknown variable '#{k}'" }
|
||||||
assigns['test'] = 'Tobi'
|
assigns['test'] = 'Tobi'
|
||||||
assert_equal 'Hello Tobi', template.render!(assigns)
|
assert_equal('Hello Tobi', template.render!(assigns))
|
||||||
assigns.delete('test')
|
assigns.delete('test')
|
||||||
e = assert_raises(RuntimeError) do
|
e = assert_raises(RuntimeError) do
|
||||||
template.render!(assigns)
|
template.render!(assigns)
|
||||||
end
|
end
|
||||||
assert_equal "Unknown variable 'test'", e.message
|
assert_equal("Unknown variable 'test'", e.message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_multiline_variable
|
def test_multiline_variable
|
||||||
assert_equal 'worked', Template.parse("{{\ntest\n}}").render!('test' => 'worked')
|
assert_equal('worked', Template.parse("{{\ntest\n}}").render!('test' => 'worked'))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_render_symbol
|
def test_render_symbol
|
||||||
assert_template_result 'bar', '{{ foo }}', 'foo' => :bar
|
assert_template_result('bar', '{{ foo }}', 'foo' => :bar)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_dynamic_find_var
|
||||||
|
assert_template_result('bar', '{{ [key] }}', 'key' => 'foo', 'foo' => 'bar')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -38,44 +38,55 @@ module Minitest
|
|||||||
include Liquid
|
include Liquid
|
||||||
|
|
||||||
def assert_template_result(expected, template, assigns = {}, message = nil)
|
def assert_template_result(expected, template, assigns = {}, message = nil)
|
||||||
assert_equal expected, Template.parse(template, line_numbers: true).render!(assigns), message
|
assert_equal(expected, Template.parse(template, line_numbers: true).render!(assigns), message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def assert_template_result_matches(expected, template, assigns = {}, message = nil)
|
def assert_template_result_matches(expected, template, assigns = {}, message = nil)
|
||||||
return assert_template_result(expected, template, assigns, message) unless expected.is_a?(Regexp)
|
return assert_template_result(expected, template, assigns, message) unless expected.is_a?(Regexp)
|
||||||
|
|
||||||
assert_match expected, Template.parse(template, line_numbers: true).render!(assigns), message
|
assert_match(expected, Template.parse(template, line_numbers: true).render!(assigns), message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def assert_match_syntax_error(match, template, assigns = {})
|
def assert_match_syntax_error(match, template, assigns = {})
|
||||||
exception = assert_raises(Liquid::SyntaxError) do
|
exception = assert_raises(Liquid::SyntaxError) do
|
||||||
Template.parse(template, line_numbers: true).render(assigns)
|
Template.parse(template, line_numbers: true).render(assigns)
|
||||||
end
|
end
|
||||||
assert_match match, exception.message
|
assert_match(match, exception.message)
|
||||||
|
end
|
||||||
|
|
||||||
|
def assert_usage_increment(name, times: 1)
|
||||||
|
old_method = Liquid::Usage.method(:increment)
|
||||||
|
calls = 0
|
||||||
|
begin
|
||||||
|
Liquid::Usage.singleton_class.send(:remove_method, :increment)
|
||||||
|
Liquid::Usage.define_singleton_method(:increment) do |got_name|
|
||||||
|
calls += 1 if got_name == name
|
||||||
|
old_method.call(got_name)
|
||||||
|
end
|
||||||
|
yield
|
||||||
|
ensure
|
||||||
|
Liquid::Usage.singleton_class.send(:remove_method, :increment)
|
||||||
|
Liquid::Usage.define_singleton_method(:increment, old_method)
|
||||||
|
end
|
||||||
|
assert_equal(times, calls, "Number of calls to Usage.increment with #{name.inspect}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def with_global_filter(*globals)
|
def with_global_filter(*globals)
|
||||||
original_global_strainer = Liquid::Strainer.class_variable_get(:@@global_strainer)
|
original_global_filters = Liquid::StrainerFactory.instance_variable_get(:@global_filters)
|
||||||
Liquid::Strainer.class_variable_set(:@@global_strainer, Class.new(Liquid::Strainer) do
|
Liquid::StrainerFactory.instance_variable_set(:@global_filters, [])
|
||||||
@filter_methods = Set.new
|
globals.each do |global|
|
||||||
end)
|
Liquid::StrainerFactory.add_global_filter(global)
|
||||||
Liquid::Strainer.class_variable_get(:@@strainer_class_cache).clear
|
end
|
||||||
|
|
||||||
|
Liquid::StrainerFactory.send(:strainer_class_cache).clear
|
||||||
|
|
||||||
globals.each do |global|
|
globals.each do |global|
|
||||||
Liquid::Template.register_filter(global)
|
Liquid::Template.register_filter(global)
|
||||||
end
|
end
|
||||||
yield
|
yield
|
||||||
ensure
|
ensure
|
||||||
Liquid::Strainer.class_variable_get(:@@strainer_class_cache).clear
|
Liquid::StrainerFactory.send(:strainer_class_cache).clear
|
||||||
Liquid::Strainer.class_variable_set(:@@global_strainer, original_global_strainer)
|
Liquid::StrainerFactory.instance_variable_set(:@global_filters, original_global_filters)
|
||||||
end
|
|
||||||
|
|
||||||
def with_taint_mode(mode)
|
|
||||||
old_mode = Liquid::Template.taint_mode
|
|
||||||
Liquid::Template.taint_mode = mode
|
|
||||||
yield
|
|
||||||
ensure
|
|
||||||
Liquid::Template.taint_mode = old_mode
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def with_error_mode(mode)
|
def with_error_mode(mode)
|
||||||
@@ -87,12 +98,19 @@ module Minitest
|
|||||||
end
|
end
|
||||||
|
|
||||||
def with_custom_tag(tag_name, tag_class)
|
def with_custom_tag(tag_name, tag_class)
|
||||||
|
old_tag = Liquid::Template.tags[tag_name]
|
||||||
|
begin
|
||||||
Liquid::Template.register_tag(tag_name, tag_class)
|
Liquid::Template.register_tag(tag_name, tag_class)
|
||||||
yield
|
yield
|
||||||
ensure
|
ensure
|
||||||
|
if old_tag
|
||||||
|
Liquid::Template.tags[tag_name] = old_tag
|
||||||
|
else
|
||||||
Liquid::Template.tags.delete(tag_name)
|
Liquid::Template.tags.delete(tag_name)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class ThingWithToLiquid
|
class ThingWithToLiquid
|
||||||
@@ -136,3 +154,16 @@ class StubFileSystem
|
|||||||
@values.fetch(template_path)
|
@values.fetch(template_path)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class StubTemplateFactory
|
||||||
|
attr_reader :count
|
||||||
|
|
||||||
|
def initialize
|
||||||
|
@count = 0
|
||||||
|
end
|
||||||
|
|
||||||
|
def for(_template_name)
|
||||||
|
@count += 1
|
||||||
|
Liquid::Template.new
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|||||||
@@ -7,86 +7,42 @@ class BlockUnitTest < Minitest::Test
|
|||||||
|
|
||||||
def test_blankspace
|
def test_blankspace
|
||||||
template = Liquid::Template.parse(" ")
|
template = Liquid::Template.parse(" ")
|
||||||
assert_equal [" "], template.root.nodelist
|
assert_equal([" "], template.root.nodelist)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_beginning
|
def test_variable_beginning
|
||||||
template = Liquid::Template.parse("{{funk}} ")
|
template = Liquid::Template.parse("{{funk}} ")
|
||||||
assert_equal 2, template.root.nodelist.size
|
assert_equal(2, template.root.nodelist.size)
|
||||||
assert_equal Variable, template.root.nodelist[0].class
|
assert_equal(Variable, template.root.nodelist[0].class)
|
||||||
assert_equal String, template.root.nodelist[1].class
|
assert_equal(String, template.root.nodelist[1].class)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_end
|
def test_variable_end
|
||||||
template = Liquid::Template.parse(" {{funk}}")
|
template = Liquid::Template.parse(" {{funk}}")
|
||||||
assert_equal 2, template.root.nodelist.size
|
assert_equal(2, template.root.nodelist.size)
|
||||||
assert_equal String, template.root.nodelist[0].class
|
assert_equal(String, template.root.nodelist[0].class)
|
||||||
assert_equal Variable, template.root.nodelist[1].class
|
assert_equal(Variable, template.root.nodelist[1].class)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_middle
|
def test_variable_middle
|
||||||
template = Liquid::Template.parse(" {{funk}} ")
|
template = Liquid::Template.parse(" {{funk}} ")
|
||||||
assert_equal 3, template.root.nodelist.size
|
assert_equal(3, template.root.nodelist.size)
|
||||||
assert_equal String, template.root.nodelist[0].class
|
assert_equal(String, template.root.nodelist[0].class)
|
||||||
assert_equal Variable, template.root.nodelist[1].class
|
assert_equal(Variable, template.root.nodelist[1].class)
|
||||||
assert_equal String, template.root.nodelist[2].class
|
assert_equal(String, template.root.nodelist[2].class)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_many_embedded_fragments
|
def test_variable_many_embedded_fragments
|
||||||
template = Liquid::Template.parse(" {{funk}} {{so}} {{brother}} ")
|
template = Liquid::Template.parse(" {{funk}} {{so}} {{brother}} ")
|
||||||
assert_equal 7, template.root.nodelist.size
|
assert_equal(7, template.root.nodelist.size)
|
||||||
assert_equal [String, Variable, String, Variable, String, Variable, String],
|
assert_equal([String, Variable, String, Variable, String, Variable, String],
|
||||||
block_types(template.root.nodelist)
|
block_types(template.root.nodelist))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_with_block
|
def test_with_block
|
||||||
template = Liquid::Template.parse(" {% comment %} {% endcomment %} ")
|
template = Liquid::Template.parse(" {% comment %} {% endcomment %} ")
|
||||||
assert_equal [String, Comment, String], block_types(template.root.nodelist)
|
assert_equal([String, Comment, String], block_types(template.root.nodelist))
|
||||||
assert_equal 3, template.root.nodelist.size
|
assert_equal(3, template.root.nodelist.size)
|
||||||
end
|
|
||||||
|
|
||||||
def test_with_custom_tag
|
|
||||||
with_custom_tag('testtag', Block) do
|
|
||||||
assert Liquid::Template.parse("{% testtag %} {% endtesttag %}")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_custom_block_tags_have_a_default_render_to_output_buffer_method_for_backwards_compatibility
|
|
||||||
klass1 = Class.new(Block) do
|
|
||||||
def render(*)
|
|
||||||
'hello'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
with_custom_tag('blabla', klass1) do
|
|
||||||
template = Liquid::Template.parse("{% blabla %} bla {% endblabla %}")
|
|
||||||
|
|
||||||
assert_equal 'hello', template.render
|
|
||||||
|
|
||||||
buf = +''
|
|
||||||
output = template.render({}, output: buf)
|
|
||||||
assert_equal 'hello', output
|
|
||||||
assert_equal 'hello', buf
|
|
||||||
assert_equal buf.object_id, output.object_id
|
|
||||||
end
|
|
||||||
|
|
||||||
klass2 = Class.new(klass1) do
|
|
||||||
def render(*)
|
|
||||||
'foo' + super + 'bar'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
with_custom_tag('blabla', klass2) do
|
|
||||||
template = Liquid::Template.parse("{% blabla %} foo {% endblabla %}")
|
|
||||||
|
|
||||||
assert_equal 'foohellobar', template.render
|
|
||||||
|
|
||||||
buf = +''
|
|
||||||
output = template.render({}, output: buf)
|
|
||||||
assert_equal 'foohellobar', output
|
|
||||||
assert_equal 'foohellobar', buf
|
|
||||||
assert_equal buf.object_id, output.object_id
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
@@ -94,4 +50,4 @@ class BlockUnitTest < Minitest::Test
|
|||||||
def block_types(nodelist)
|
def block_types(nodelist)
|
||||||
nodelist.collect(&:class)
|
nodelist.collect(&:class)
|
||||||
end
|
end
|
||||||
end # VariableTest
|
end
|
||||||
|
|||||||
@@ -10,68 +10,68 @@ class ConditionUnitTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_basic_condition
|
def test_basic_condition
|
||||||
assert_equal false, Condition.new(1, '==', 2).evaluate
|
assert_equal(false, Condition.new(1, '==', 2).evaluate)
|
||||||
assert_equal true, Condition.new(1, '==', 1).evaluate
|
assert_equal(true, Condition.new(1, '==', 1).evaluate)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_default_operators_evalute_true
|
def test_default_operators_evalute_true
|
||||||
assert_evaluates_true 1, '==', 1
|
assert_evaluates_true(1, '==', 1)
|
||||||
assert_evaluates_true 1, '!=', 2
|
assert_evaluates_true(1, '!=', 2)
|
||||||
assert_evaluates_true 1, '<>', 2
|
assert_evaluates_true(1, '<>', 2)
|
||||||
assert_evaluates_true 1, '<', 2
|
assert_evaluates_true(1, '<', 2)
|
||||||
assert_evaluates_true 2, '>', 1
|
assert_evaluates_true(2, '>', 1)
|
||||||
assert_evaluates_true 1, '>=', 1
|
assert_evaluates_true(1, '>=', 1)
|
||||||
assert_evaluates_true 2, '>=', 1
|
assert_evaluates_true(2, '>=', 1)
|
||||||
assert_evaluates_true 1, '<=', 2
|
assert_evaluates_true(1, '<=', 2)
|
||||||
assert_evaluates_true 1, '<=', 1
|
assert_evaluates_true(1, '<=', 1)
|
||||||
# negative numbers
|
# negative numbers
|
||||||
assert_evaluates_true 1, '>', -1
|
assert_evaluates_true(1, '>', -1)
|
||||||
assert_evaluates_true(-1, '<', 1)
|
assert_evaluates_true(-1, '<', 1)
|
||||||
assert_evaluates_true 1.0, '>', -1.0
|
assert_evaluates_true(1.0, '>', -1.0)
|
||||||
assert_evaluates_true(-1.0, '<', 1.0)
|
assert_evaluates_true(-1.0, '<', 1.0)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_default_operators_evalute_false
|
def test_default_operators_evalute_false
|
||||||
assert_evaluates_false 1, '==', 2
|
assert_evaluates_false(1, '==', 2)
|
||||||
assert_evaluates_false 1, '!=', 1
|
assert_evaluates_false(1, '!=', 1)
|
||||||
assert_evaluates_false 1, '<>', 1
|
assert_evaluates_false(1, '<>', 1)
|
||||||
assert_evaluates_false 1, '<', 0
|
assert_evaluates_false(1, '<', 0)
|
||||||
assert_evaluates_false 2, '>', 4
|
assert_evaluates_false(2, '>', 4)
|
||||||
assert_evaluates_false 1, '>=', 3
|
assert_evaluates_false(1, '>=', 3)
|
||||||
assert_evaluates_false 2, '>=', 4
|
assert_evaluates_false(2, '>=', 4)
|
||||||
assert_evaluates_false 1, '<=', 0
|
assert_evaluates_false(1, '<=', 0)
|
||||||
assert_evaluates_false 1, '<=', 0
|
assert_evaluates_false(1, '<=', 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_contains_works_on_strings
|
def test_contains_works_on_strings
|
||||||
assert_evaluates_true 'bob', 'contains', 'o'
|
assert_evaluates_true('bob', 'contains', 'o')
|
||||||
assert_evaluates_true 'bob', 'contains', 'b'
|
assert_evaluates_true('bob', 'contains', 'b')
|
||||||
assert_evaluates_true 'bob', 'contains', 'bo'
|
assert_evaluates_true('bob', 'contains', 'bo')
|
||||||
assert_evaluates_true 'bob', 'contains', 'ob'
|
assert_evaluates_true('bob', 'contains', 'ob')
|
||||||
assert_evaluates_true 'bob', 'contains', 'bob'
|
assert_evaluates_true('bob', 'contains', 'bob')
|
||||||
|
|
||||||
assert_evaluates_false 'bob', 'contains', 'bob2'
|
assert_evaluates_false('bob', 'contains', 'bob2')
|
||||||
assert_evaluates_false 'bob', 'contains', 'a'
|
assert_evaluates_false('bob', 'contains', 'a')
|
||||||
assert_evaluates_false 'bob', 'contains', '---'
|
assert_evaluates_false('bob', 'contains', '---')
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_invalid_comparation_operator
|
def test_invalid_comparation_operator
|
||||||
assert_evaluates_argument_error 1, '~~', 0
|
assert_evaluates_argument_error(1, '~~', 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_comparation_of_int_and_str
|
def test_comparation_of_int_and_str
|
||||||
assert_evaluates_argument_error '1', '>', 0
|
assert_evaluates_argument_error('1', '>', 0)
|
||||||
assert_evaluates_argument_error '1', '<', 0
|
assert_evaluates_argument_error('1', '<', 0)
|
||||||
assert_evaluates_argument_error '1', '>=', 0
|
assert_evaluates_argument_error('1', '>=', 0)
|
||||||
assert_evaluates_argument_error '1', '<=', 0
|
assert_evaluates_argument_error('1', '<=', 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_hash_compare_backwards_compatibility
|
def test_hash_compare_backwards_compatibility
|
||||||
assert_nil Condition.new({}, '>', 2).evaluate
|
assert_nil(Condition.new({}, '>', 2).evaluate)
|
||||||
assert_nil Condition.new(2, '>', {}).evaluate
|
assert_nil(Condition.new(2, '>', {}).evaluate)
|
||||||
assert_equal false, Condition.new({}, '==', 2).evaluate
|
assert_equal(false, Condition.new({}, '==', 2).evaluate)
|
||||||
assert_equal true, Condition.new({ 'a' => 1 }, '==', 'a' => 1).evaluate
|
assert_equal(true, Condition.new({ 'a' => 1 }, '==', 'a' => 1).evaluate)
|
||||||
assert_equal true, Condition.new({ 'a' => 2 }, 'contains', 'a').evaluate
|
assert_equal(true, Condition.new({ 'a' => 2 }, 'contains', 'a').evaluate)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_contains_works_on_arrays
|
def test_contains_works_on_arrays
|
||||||
@@ -79,64 +79,64 @@ class ConditionUnitTest < Minitest::Test
|
|||||||
@context['array'] = [1, 2, 3, 4, 5]
|
@context['array'] = [1, 2, 3, 4, 5]
|
||||||
array_expr = VariableLookup.new("array")
|
array_expr = VariableLookup.new("array")
|
||||||
|
|
||||||
assert_evaluates_false array_expr, 'contains', 0
|
assert_evaluates_false(array_expr, 'contains', 0)
|
||||||
assert_evaluates_true array_expr, 'contains', 1
|
assert_evaluates_true(array_expr, 'contains', 1)
|
||||||
assert_evaluates_true array_expr, 'contains', 2
|
assert_evaluates_true(array_expr, 'contains', 2)
|
||||||
assert_evaluates_true array_expr, 'contains', 3
|
assert_evaluates_true(array_expr, 'contains', 3)
|
||||||
assert_evaluates_true array_expr, 'contains', 4
|
assert_evaluates_true(array_expr, 'contains', 4)
|
||||||
assert_evaluates_true array_expr, 'contains', 5
|
assert_evaluates_true(array_expr, 'contains', 5)
|
||||||
assert_evaluates_false array_expr, 'contains', 6
|
assert_evaluates_false(array_expr, 'contains', 6)
|
||||||
assert_evaluates_false array_expr, 'contains', "1"
|
assert_evaluates_false(array_expr, 'contains', "1")
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_contains_returns_false_for_nil_operands
|
def test_contains_returns_false_for_nil_operands
|
||||||
@context = Liquid::Context.new
|
@context = Liquid::Context.new
|
||||||
assert_evaluates_false VariableLookup.new('not_assigned'), 'contains', '0'
|
assert_evaluates_false(VariableLookup.new('not_assigned'), 'contains', '0')
|
||||||
assert_evaluates_false 0, 'contains', VariableLookup.new('not_assigned')
|
assert_evaluates_false(0, 'contains', VariableLookup.new('not_assigned'))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_contains_return_false_on_wrong_data_type
|
def test_contains_return_false_on_wrong_data_type
|
||||||
assert_evaluates_false 1, 'contains', 0
|
assert_evaluates_false(1, 'contains', 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_contains_with_string_left_operand_coerces_right_operand_to_string
|
def test_contains_with_string_left_operand_coerces_right_operand_to_string
|
||||||
assert_evaluates_true ' 1 ', 'contains', 1
|
assert_evaluates_true(' 1 ', 'contains', 1)
|
||||||
assert_evaluates_false ' 1 ', 'contains', 2
|
assert_evaluates_false(' 1 ', 'contains', 2)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_or_condition
|
def test_or_condition
|
||||||
condition = Condition.new(1, '==', 2)
|
condition = Condition.new(1, '==', 2)
|
||||||
|
|
||||||
assert_equal false, condition.evaluate
|
assert_equal(false, condition.evaluate)
|
||||||
|
|
||||||
condition.or(Condition.new(2, '==', 1))
|
condition.or(Condition.new(2, '==', 1))
|
||||||
|
|
||||||
assert_equal false, condition.evaluate
|
assert_equal(false, condition.evaluate)
|
||||||
|
|
||||||
condition.or(Condition.new(1, '==', 1))
|
condition.or(Condition.new(1, '==', 1))
|
||||||
|
|
||||||
assert_equal true, condition.evaluate
|
assert_equal(true, condition.evaluate)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_and_condition
|
def test_and_condition
|
||||||
condition = Condition.new(1, '==', 1)
|
condition = Condition.new(1, '==', 1)
|
||||||
|
|
||||||
assert_equal true, condition.evaluate
|
assert_equal(true, condition.evaluate)
|
||||||
|
|
||||||
condition.and(Condition.new(2, '==', 2))
|
condition.and(Condition.new(2, '==', 2))
|
||||||
|
|
||||||
assert_equal true, condition.evaluate
|
assert_equal(true, condition.evaluate)
|
||||||
|
|
||||||
condition.and(Condition.new(2, '==', 1))
|
condition.and(Condition.new(2, '==', 1))
|
||||||
|
|
||||||
assert_equal false, condition.evaluate
|
assert_equal(false, condition.evaluate)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_should_allow_custom_proc_operator
|
def test_should_allow_custom_proc_operator
|
||||||
Condition.operators['starts_with'] = proc { |_cond, left, right| left =~ /^#{right}/ }
|
Condition.operators['starts_with'] = proc { |_cond, left, right| left =~ /^#{right}/ }
|
||||||
|
|
||||||
assert_evaluates_true 'bob', 'starts_with', 'b'
|
assert_evaluates_true('bob', 'starts_with', 'b')
|
||||||
assert_evaluates_false 'bob', 'starts_with', 'o'
|
assert_evaluates_false('bob', 'starts_with', 'o')
|
||||||
ensure
|
ensure
|
||||||
Condition.operators.delete('starts_with')
|
Condition.operators.delete('starts_with')
|
||||||
end
|
end
|
||||||
@@ -145,19 +145,19 @@ class ConditionUnitTest < Minitest::Test
|
|||||||
@context = Liquid::Context.new
|
@context = Liquid::Context.new
|
||||||
@context['one'] = @context['another'] = "gnomeslab-and-or-liquid"
|
@context['one'] = @context['another'] = "gnomeslab-and-or-liquid"
|
||||||
|
|
||||||
assert_evaluates_true VariableLookup.new("one"), '==', VariableLookup.new("another")
|
assert_evaluates_true(VariableLookup.new("one"), '==', VariableLookup.new("another"))
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def assert_evaluates_true(left, op, right)
|
def assert_evaluates_true(left, op, right)
|
||||||
assert Condition.new(left, op, right).evaluate(@context),
|
assert(Condition.new(left, op, right).evaluate(@context),
|
||||||
"Evaluated false: #{left} #{op} #{right}"
|
"Evaluated false: #{left} #{op} #{right}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def assert_evaluates_false(left, op, right)
|
def assert_evaluates_false(left, op, right)
|
||||||
assert !Condition.new(left, op, right).evaluate(@context),
|
assert(!Condition.new(left, op, right).evaluate(@context),
|
||||||
"Evaluated true: #{left} #{op} #{right}"
|
"Evaluated true: #{left} #{op} #{right}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def assert_evaluates_argument_error(left, op, right)
|
def assert_evaluates_argument_error(left, op, right)
|
||||||
|
|||||||
@@ -1,579 +0,0 @@
|
|||||||
# frozen_string_literal: true
|
|
||||||
|
|
||||||
require 'test_helper'
|
|
||||||
|
|
||||||
class HundredCentes
|
|
||||||
def to_liquid
|
|
||||||
100
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class CentsDrop < Liquid::Drop
|
|
||||||
def amount
|
|
||||||
HundredCentes.new
|
|
||||||
end
|
|
||||||
|
|
||||||
def non_zero?
|
|
||||||
true
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class ContextSensitiveDrop < Liquid::Drop
|
|
||||||
def test
|
|
||||||
@context['test']
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class Category < Liquid::Drop
|
|
||||||
attr_accessor :name
|
|
||||||
|
|
||||||
def initialize(name)
|
|
||||||
@name = name
|
|
||||||
end
|
|
||||||
|
|
||||||
def to_liquid
|
|
||||||
CategoryDrop.new(self)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class CategoryDrop
|
|
||||||
attr_accessor :category, :context
|
|
||||||
def initialize(category)
|
|
||||||
@category = category
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class CounterDrop < Liquid::Drop
|
|
||||||
def count
|
|
||||||
@count ||= 0
|
|
||||||
@count += 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class ArrayLike
|
|
||||||
def fetch(index)
|
|
||||||
end
|
|
||||||
|
|
||||||
def [](index)
|
|
||||||
@counts ||= []
|
|
||||||
@counts[index] ||= 0
|
|
||||||
@counts[index] += 1
|
|
||||||
end
|
|
||||||
|
|
||||||
def to_liquid
|
|
||||||
self
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
class ContextUnitTest < Minitest::Test
|
|
||||||
include Liquid
|
|
||||||
|
|
||||||
def setup
|
|
||||||
@context = Liquid::Context.new
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_variables
|
|
||||||
@context['string'] = 'string'
|
|
||||||
assert_equal 'string', @context['string']
|
|
||||||
|
|
||||||
@context['num'] = 5
|
|
||||||
assert_equal 5, @context['num']
|
|
||||||
|
|
||||||
@context['time'] = Time.parse('2006-06-06 12:00:00')
|
|
||||||
assert_equal Time.parse('2006-06-06 12:00:00'), @context['time']
|
|
||||||
|
|
||||||
@context['date'] = Date.today
|
|
||||||
assert_equal Date.today, @context['date']
|
|
||||||
|
|
||||||
now = Time.now
|
|
||||||
@context['datetime'] = now
|
|
||||||
assert_equal now, @context['datetime']
|
|
||||||
|
|
||||||
@context['bool'] = true
|
|
||||||
assert_equal true, @context['bool']
|
|
||||||
|
|
||||||
@context['bool'] = false
|
|
||||||
assert_equal false, @context['bool']
|
|
||||||
|
|
||||||
@context['nil'] = nil
|
|
||||||
assert_nil @context['nil']
|
|
||||||
assert_nil @context['nil']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_variables_not_existing
|
|
||||||
assert_nil @context['does_not_exist']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_scoping
|
|
||||||
@context.push
|
|
||||||
@context.pop
|
|
||||||
|
|
||||||
assert_raises(Liquid::ContextError) do
|
|
||||||
@context.pop
|
|
||||||
end
|
|
||||||
|
|
||||||
assert_raises(Liquid::ContextError) do
|
|
||||||
@context.push
|
|
||||||
@context.pop
|
|
||||||
@context.pop
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_length_query
|
|
||||||
@context['numbers'] = [1, 2, 3, 4]
|
|
||||||
|
|
||||||
assert_equal 4, @context['numbers.size']
|
|
||||||
|
|
||||||
@context['numbers'] = { 1 => 1, 2 => 2, 3 => 3, 4 => 4 }
|
|
||||||
|
|
||||||
assert_equal 4, @context['numbers.size']
|
|
||||||
|
|
||||||
@context['numbers'] = { 1 => 1, 2 => 2, 3 => 3, 4 => 4, 'size' => 1000 }
|
|
||||||
|
|
||||||
assert_equal 1000, @context['numbers.size']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_hyphenated_variable
|
|
||||||
@context['oh-my'] = 'godz'
|
|
||||||
assert_equal 'godz', @context['oh-my']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_add_filter
|
|
||||||
filter = Module.new do
|
|
||||||
def hi(output)
|
|
||||||
output + ' hi!'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
context = Context.new
|
|
||||||
context.add_filters(filter)
|
|
||||||
assert_equal 'hi? hi!', context.invoke(:hi, 'hi?')
|
|
||||||
|
|
||||||
context = Context.new
|
|
||||||
assert_equal 'hi?', context.invoke(:hi, 'hi?')
|
|
||||||
|
|
||||||
context.add_filters(filter)
|
|
||||||
assert_equal 'hi? hi!', context.invoke(:hi, 'hi?')
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_only_intended_filters_make_it_there
|
|
||||||
filter = Module.new do
|
|
||||||
def hi(output)
|
|
||||||
output + ' hi!'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
context = Context.new
|
|
||||||
assert_equal "Wookie", context.invoke("hi", "Wookie")
|
|
||||||
|
|
||||||
context.add_filters(filter)
|
|
||||||
assert_equal "Wookie hi!", context.invoke("hi", "Wookie")
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_add_item_in_outer_scope
|
|
||||||
@context['test'] = 'test'
|
|
||||||
@context.push
|
|
||||||
assert_equal 'test', @context['test']
|
|
||||||
@context.pop
|
|
||||||
assert_equal 'test', @context['test']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_add_item_in_inner_scope
|
|
||||||
@context.push
|
|
||||||
@context['test'] = 'test'
|
|
||||||
assert_equal 'test', @context['test']
|
|
||||||
@context.pop
|
|
||||||
assert_nil @context['test']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_hierachical_data
|
|
||||||
@context['hash'] = { "name" => 'tobi' }
|
|
||||||
assert_equal 'tobi', @context['hash.name']
|
|
||||||
assert_equal 'tobi', @context['hash["name"]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_keywords
|
|
||||||
assert_equal true, @context['true']
|
|
||||||
assert_equal false, @context['false']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_digits
|
|
||||||
assert_equal 100, @context['100']
|
|
||||||
assert_equal 100.00, @context['100.00']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_strings
|
|
||||||
assert_equal "hello!", @context['"hello!"']
|
|
||||||
assert_equal "hello!", @context["'hello!'"]
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_merge
|
|
||||||
@context.merge("test" => "test")
|
|
||||||
assert_equal 'test', @context['test']
|
|
||||||
@context.merge("test" => "newvalue", "foo" => "bar")
|
|
||||||
assert_equal 'newvalue', @context['test']
|
|
||||||
assert_equal 'bar', @context['foo']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_array_notation
|
|
||||||
@context['test'] = [1, 2, 3, 4, 5]
|
|
||||||
|
|
||||||
assert_equal 1, @context['test[0]']
|
|
||||||
assert_equal 2, @context['test[1]']
|
|
||||||
assert_equal 3, @context['test[2]']
|
|
||||||
assert_equal 4, @context['test[3]']
|
|
||||||
assert_equal 5, @context['test[4]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_recoursive_array_notation
|
|
||||||
@context['test'] = { 'test' => [1, 2, 3, 4, 5] }
|
|
||||||
|
|
||||||
assert_equal 1, @context['test.test[0]']
|
|
||||||
|
|
||||||
@context['test'] = [{ 'test' => 'worked' }]
|
|
||||||
|
|
||||||
assert_equal 'worked', @context['test[0].test']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_hash_to_array_transition
|
|
||||||
@context['colors'] = {
|
|
||||||
'Blue' => ['003366', '336699', '6699CC', '99CCFF'],
|
|
||||||
'Green' => ['003300', '336633', '669966', '99CC99'],
|
|
||||||
'Yellow' => ['CC9900', 'FFCC00', 'FFFF99', 'FFFFCC'],
|
|
||||||
'Red' => ['660000', '993333', 'CC6666', 'FF9999'],
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_equal '003366', @context['colors.Blue[0]']
|
|
||||||
assert_equal 'FF9999', @context['colors.Red[3]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_try_first
|
|
||||||
@context['test'] = [1, 2, 3, 4, 5]
|
|
||||||
|
|
||||||
assert_equal 1, @context['test.first']
|
|
||||||
assert_equal 5, @context['test.last']
|
|
||||||
|
|
||||||
@context['test'] = { 'test' => [1, 2, 3, 4, 5] }
|
|
||||||
|
|
||||||
assert_equal 1, @context['test.test.first']
|
|
||||||
assert_equal 5, @context['test.test.last']
|
|
||||||
|
|
||||||
@context['test'] = [1]
|
|
||||||
assert_equal 1, @context['test.first']
|
|
||||||
assert_equal 1, @context['test.last']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_access_hashes_with_hash_notation
|
|
||||||
@context['products'] = { 'count' => 5, 'tags' => ['deepsnow', 'freestyle'] }
|
|
||||||
@context['product'] = { 'variants' => [{ 'title' => 'draft151cm' }, { 'title' => 'element151cm' }] }
|
|
||||||
|
|
||||||
assert_equal 5, @context['products["count"]']
|
|
||||||
assert_equal 'deepsnow', @context['products["tags"][0]']
|
|
||||||
assert_equal 'deepsnow', @context['products["tags"].first']
|
|
||||||
assert_equal 'draft151cm', @context['product["variants"][0]["title"]']
|
|
||||||
assert_equal 'element151cm', @context['product["variants"][1]["title"]']
|
|
||||||
assert_equal 'draft151cm', @context['product["variants"][0]["title"]']
|
|
||||||
assert_equal 'element151cm', @context['product["variants"].last["title"]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_access_variable_with_hash_notation
|
|
||||||
@context['foo'] = 'baz'
|
|
||||||
@context['bar'] = 'foo'
|
|
||||||
|
|
||||||
assert_equal 'baz', @context['["foo"]']
|
|
||||||
assert_equal 'baz', @context['[bar]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_access_hashes_with_hash_access_variables
|
|
||||||
@context['var'] = 'tags'
|
|
||||||
@context['nested'] = { 'var' => 'tags' }
|
|
||||||
@context['products'] = { 'count' => 5, 'tags' => ['deepsnow', 'freestyle'] }
|
|
||||||
|
|
||||||
assert_equal 'deepsnow', @context['products[var].first']
|
|
||||||
assert_equal 'freestyle', @context['products[nested.var].last']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_hash_notation_only_for_hash_access
|
|
||||||
@context['array'] = [1, 2, 3, 4, 5]
|
|
||||||
@context['hash'] = { 'first' => 'Hello' }
|
|
||||||
|
|
||||||
assert_equal 1, @context['array.first']
|
|
||||||
assert_nil @context['array["first"]']
|
|
||||||
assert_equal 'Hello', @context['hash["first"]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_first_can_appear_in_middle_of_callchain
|
|
||||||
@context['product'] = { 'variants' => [{ 'title' => 'draft151cm' }, { 'title' => 'element151cm' }] }
|
|
||||||
|
|
||||||
assert_equal 'draft151cm', @context['product.variants[0].title']
|
|
||||||
assert_equal 'element151cm', @context['product.variants[1].title']
|
|
||||||
assert_equal 'draft151cm', @context['product.variants.first.title']
|
|
||||||
assert_equal 'element151cm', @context['product.variants.last.title']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_cents
|
|
||||||
@context.merge("cents" => HundredCentes.new)
|
|
||||||
assert_equal 100, @context['cents']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_nested_cents
|
|
||||||
@context.merge("cents" => { 'amount' => HundredCentes.new })
|
|
||||||
assert_equal 100, @context['cents.amount']
|
|
||||||
|
|
||||||
@context.merge("cents" => { 'cents' => { 'amount' => HundredCentes.new } })
|
|
||||||
assert_equal 100, @context['cents.cents.amount']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_cents_through_drop
|
|
||||||
@context.merge("cents" => CentsDrop.new)
|
|
||||||
assert_equal 100, @context['cents.amount']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_nested_cents_through_drop
|
|
||||||
@context.merge("vars" => { "cents" => CentsDrop.new })
|
|
||||||
assert_equal 100, @context['vars.cents.amount']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_drop_methods_with_question_marks
|
|
||||||
@context.merge("cents" => CentsDrop.new)
|
|
||||||
assert @context['cents.non_zero?']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_context_from_within_drop
|
|
||||||
@context.merge("test" => '123', "vars" => ContextSensitiveDrop.new)
|
|
||||||
assert_equal '123', @context['vars.test']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_nested_context_from_within_drop
|
|
||||||
@context.merge("test" => '123', "vars" => { "local" => ContextSensitiveDrop.new })
|
|
||||||
assert_equal '123', @context['vars.local.test']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_ranges
|
|
||||||
@context.merge("test" => '5')
|
|
||||||
assert_equal (1..5), @context['(1..5)']
|
|
||||||
assert_equal (1..5), @context['(1..test)']
|
|
||||||
assert_equal (5..5), @context['(test..test)']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_cents_through_drop_nestedly
|
|
||||||
@context.merge("cents" => { "cents" => CentsDrop.new })
|
|
||||||
assert_equal 100, @context['cents.cents.amount']
|
|
||||||
|
|
||||||
@context.merge("cents" => { "cents" => { "cents" => CentsDrop.new } })
|
|
||||||
assert_equal 100, @context['cents.cents.cents.amount']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_drop_with_variable_called_only_once
|
|
||||||
@context['counter'] = CounterDrop.new
|
|
||||||
|
|
||||||
assert_equal 1, @context['counter.count']
|
|
||||||
assert_equal 2, @context['counter.count']
|
|
||||||
assert_equal 3, @context['counter.count']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_drop_with_key_called_only_once
|
|
||||||
@context['counter'] = CounterDrop.new
|
|
||||||
|
|
||||||
assert_equal 1, @context['counter["count"]']
|
|
||||||
assert_equal 2, @context['counter["count"]']
|
|
||||||
assert_equal 3, @context['counter["count"]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_proc_as_variable
|
|
||||||
@context['dynamic'] = proc { 'Hello' }
|
|
||||||
|
|
||||||
assert_equal 'Hello', @context['dynamic']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_lambda_as_variable
|
|
||||||
@context['dynamic'] = proc { 'Hello' }
|
|
||||||
|
|
||||||
assert_equal 'Hello', @context['dynamic']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_nested_lambda_as_variable
|
|
||||||
@context['dynamic'] = { "lambda" => proc { 'Hello' } }
|
|
||||||
|
|
||||||
assert_equal 'Hello', @context['dynamic.lambda']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_array_containing_lambda_as_variable
|
|
||||||
@context['dynamic'] = [1, 2, proc { 'Hello' }, 4, 5]
|
|
||||||
|
|
||||||
assert_equal 'Hello', @context['dynamic[2]']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_lambda_is_called_once
|
|
||||||
@context['callcount'] = proc {
|
|
||||||
@global ||= 0
|
|
||||||
@global += 1
|
|
||||||
@global.to_s
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_equal '1', @context['callcount']
|
|
||||||
assert_equal '1', @context['callcount']
|
|
||||||
assert_equal '1', @context['callcount']
|
|
||||||
|
|
||||||
@global = nil
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_nested_lambda_is_called_once
|
|
||||||
@context['callcount'] = { "lambda" => proc {
|
|
||||||
@global ||= 0
|
|
||||||
@global += 1
|
|
||||||
@global.to_s
|
|
||||||
} }
|
|
||||||
|
|
||||||
assert_equal '1', @context['callcount.lambda']
|
|
||||||
assert_equal '1', @context['callcount.lambda']
|
|
||||||
assert_equal '1', @context['callcount.lambda']
|
|
||||||
|
|
||||||
@global = nil
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_lambda_in_array_is_called_once
|
|
||||||
@context['callcount'] = [1, 2, proc {
|
|
||||||
@global ||= 0
|
|
||||||
@global += 1
|
|
||||||
@global.to_s
|
|
||||||
}, 4, 5]
|
|
||||||
|
|
||||||
assert_equal '1', @context['callcount[2]']
|
|
||||||
assert_equal '1', @context['callcount[2]']
|
|
||||||
assert_equal '1', @context['callcount[2]']
|
|
||||||
|
|
||||||
@global = nil
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_access_to_context_from_proc
|
|
||||||
@context.registers[:magic] = 345392
|
|
||||||
|
|
||||||
@context['magic'] = proc { @context.registers[:magic] }
|
|
||||||
|
|
||||||
assert_equal 345392, @context['magic']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_to_liquid_and_context_at_first_level
|
|
||||||
@context['category'] = Category.new("foobar")
|
|
||||||
assert_kind_of CategoryDrop, @context['category']
|
|
||||||
assert_equal @context, @context['category'].context
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_interrupt_avoids_object_allocations
|
|
||||||
assert_no_object_allocations do
|
|
||||||
@context.interrupt?
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_context_initialization_with_a_proc_in_environment
|
|
||||||
contx = Context.new([test: ->(c) { c['poutine'] }], test: :foo)
|
|
||||||
|
|
||||||
assert contx
|
|
||||||
assert_nil contx['poutine']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_apply_global_filter
|
|
||||||
global_filter_proc = ->(output) { "#{output} filtered" }
|
|
||||||
|
|
||||||
context = Context.new
|
|
||||||
context.global_filter = global_filter_proc
|
|
||||||
|
|
||||||
assert_equal 'hi filtered', context.apply_global_filter('hi')
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_static_environments_are_read_with_lower_priority_than_environments
|
|
||||||
context = Context.build(
|
|
||||||
static_environments: { 'shadowed' => 'static', 'unshadowed' => 'static' },
|
|
||||||
environments: { 'shadowed' => 'dynamic' }
|
|
||||||
)
|
|
||||||
|
|
||||||
assert_equal 'dynamic', context['shadowed']
|
|
||||||
assert_equal 'static', context['unshadowed']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_apply_global_filter_when_no_global_filter_exist
|
|
||||||
context = Context.new
|
|
||||||
assert_equal 'hi', context.apply_global_filter('hi')
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_does_not_inherit_variables
|
|
||||||
super_context = Context.new
|
|
||||||
super_context['my_variable'] = 'some value'
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
|
|
||||||
assert_nil subcontext['my_variable']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_inherits_static_environment
|
|
||||||
super_context = Context.build(static_environments: { 'my_environment_value' => 'my value' })
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
|
|
||||||
assert_equal 'my value', subcontext['my_environment_value']
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_inherits_resource_limits
|
|
||||||
resource_limits = ResourceLimits.new({})
|
|
||||||
super_context = Context.new({}, {}, {}, false, resource_limits)
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
assert_equal resource_limits, subcontext.resource_limits
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_inherits_exception_renderer
|
|
||||||
super_context = Context.new
|
|
||||||
super_context.exception_renderer = ->(_e) { 'my exception message' }
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
assert_equal 'my exception message', subcontext.handle_error(Liquid::Error.new)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_does_not_inherit_non_static_registers
|
|
||||||
registers = {
|
|
||||||
my_register: :my_value,
|
|
||||||
}
|
|
||||||
super_context = Context.new({}, {}, StaticRegisters.new(registers))
|
|
||||||
super_context.registers[:my_register] = :my_alt_value
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
assert_equal :my_value, subcontext.registers[:my_register]
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_inherits_static_registers
|
|
||||||
super_context = Context.build(registers: { my_register: :my_value })
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
assert_equal :my_value, subcontext.registers[:my_register]
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_registers_do_not_pollute_context
|
|
||||||
super_context = Context.build(registers: { my_register: :my_value })
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
subcontext.registers[:my_register] = :my_alt_value
|
|
||||||
assert_equal :my_value, super_context.registers[:my_register]
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_new_isolated_subcontext_inherits_filters
|
|
||||||
my_filter = Module.new do
|
|
||||||
def my_filter(*)
|
|
||||||
'my filter result'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
super_context = Context.new
|
|
||||||
super_context.add_filters([my_filter])
|
|
||||||
subcontext = super_context.new_isolated_subcontext
|
|
||||||
template = Template.parse('{{ 123 | my_filter }}')
|
|
||||||
assert_equal 'my filter result', template.render(subcontext)
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def assert_no_object_allocations
|
|
||||||
unless RUBY_ENGINE == 'ruby'
|
|
||||||
skip "stackprof needed to count object allocations"
|
|
||||||
end
|
|
||||||
require 'stackprof'
|
|
||||||
|
|
||||||
profile = StackProf.run(mode: :object) do
|
|
||||||
yield
|
|
||||||
end
|
|
||||||
assert_equal 0, profile[:samples]
|
|
||||||
end
|
|
||||||
end # ContextTest
|
|
||||||
@@ -13,8 +13,8 @@ class FileSystemUnitTest < Minitest::Test
|
|||||||
|
|
||||||
def test_local
|
def test_local
|
||||||
file_system = Liquid::LocalFileSystem.new("/some/path")
|
file_system = Liquid::LocalFileSystem.new("/some/path")
|
||||||
assert_equal "/some/path/_mypartial.liquid", file_system.full_path("mypartial")
|
assert_equal("/some/path/_mypartial.liquid", file_system.full_path("mypartial"))
|
||||||
assert_equal "/some/path/dir/_mypartial.liquid", file_system.full_path("dir/mypartial")
|
assert_equal("/some/path/dir/_mypartial.liquid", file_system.full_path("dir/mypartial"))
|
||||||
|
|
||||||
assert_raises(FileSystemError) do
|
assert_raises(FileSystemError) do
|
||||||
file_system.full_path("../dir/mypartial")
|
file_system.full_path("../dir/mypartial")
|
||||||
@@ -31,7 +31,7 @@ class FileSystemUnitTest < Minitest::Test
|
|||||||
|
|
||||||
def test_custom_template_filename_patterns
|
def test_custom_template_filename_patterns
|
||||||
file_system = Liquid::LocalFileSystem.new("/some/path", "%s.html")
|
file_system = Liquid::LocalFileSystem.new("/some/path", "%s.html")
|
||||||
assert_equal "/some/path/mypartial.html", file_system.full_path("mypartial")
|
assert_equal("/some/path/mypartial.html", file_system.full_path("mypartial"))
|
||||||
assert_equal "/some/path/dir/mypartial.html", file_system.full_path("dir/mypartial")
|
assert_equal("/some/path/dir/mypartial.html", file_system.full_path("dir/mypartial"))
|
||||||
end
|
end
|
||||||
end # FileSystemTest
|
end # FileSystemTest
|
||||||
|
|||||||
@@ -10,15 +10,15 @@ class I18nUnitTest < Minitest::Test
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_simple_translate_string
|
def test_simple_translate_string
|
||||||
assert_equal "less is more", @i18n.translate("simple")
|
assert_equal("less is more", @i18n.translate("simple"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_nested_translate_string
|
def test_nested_translate_string
|
||||||
assert_equal "something wasn't right", @i18n.translate("errors.syntax.oops")
|
assert_equal("something wasn't right", @i18n.translate("errors.syntax.oops"))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_single_string_interpolation
|
def test_single_string_interpolation
|
||||||
assert_equal "something different", @i18n.translate("whatever", something: "different")
|
assert_equal("something different", @i18n.translate("whatever", something: "different"))
|
||||||
end
|
end
|
||||||
|
|
||||||
# def test_raises_translation_error_on_undefined_interpolation_key
|
# def test_raises_translation_error_on_undefined_interpolation_key
|
||||||
@@ -28,12 +28,12 @@ class I18nUnitTest < Minitest::Test
|
|||||||
# end
|
# end
|
||||||
|
|
||||||
def test_raises_unknown_translation
|
def test_raises_unknown_translation
|
||||||
assert_raises I18n::TranslationError do
|
assert_raises(I18n::TranslationError) do
|
||||||
@i18n.translate("doesnt_exist")
|
@i18n.translate("doesnt_exist")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_sets_default_path_to_en
|
def test_sets_default_path_to_en
|
||||||
assert_equal I18n::DEFAULT_LOCALE, I18n.new.path
|
assert_equal(I18n::DEFAULT_LOCALE, I18n.new.path)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -7,42 +7,42 @@ class LexerUnitTest < Minitest::Test
|
|||||||
|
|
||||||
def test_strings
|
def test_strings
|
||||||
tokens = Lexer.new(%( 'this is a test""' "wat 'lol'")).tokenize
|
tokens = Lexer.new(%( 'this is a test""' "wat 'lol'")).tokenize
|
||||||
assert_equal [[:string, %('this is a test""')], [:string, %("wat 'lol'")], [:end_of_string]], tokens
|
assert_equal([[:string, %('this is a test""')], [:string, %("wat 'lol'")], [:end_of_string]], tokens)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_integer
|
def test_integer
|
||||||
tokens = Lexer.new('hi 50').tokenize
|
tokens = Lexer.new('hi 50').tokenize
|
||||||
assert_equal [[:id, 'hi'], [:number, '50'], [:end_of_string]], tokens
|
assert_equal([[:id, 'hi'], [:number, '50'], [:end_of_string]], tokens)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_float
|
def test_float
|
||||||
tokens = Lexer.new('hi 5.0').tokenize
|
tokens = Lexer.new('hi 5.0').tokenize
|
||||||
assert_equal [[:id, 'hi'], [:number, '5.0'], [:end_of_string]], tokens
|
assert_equal([[:id, 'hi'], [:number, '5.0'], [:end_of_string]], tokens)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_comparison
|
def test_comparison
|
||||||
tokens = Lexer.new('== <> contains ').tokenize
|
tokens = Lexer.new('== <> contains ').tokenize
|
||||||
assert_equal [[:comparison, '=='], [:comparison, '<>'], [:comparison, 'contains'], [:end_of_string]], tokens
|
assert_equal([[:comparison, '=='], [:comparison, '<>'], [:comparison, 'contains'], [:end_of_string]], tokens)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_specials
|
def test_specials
|
||||||
tokens = Lexer.new('| .:').tokenize
|
tokens = Lexer.new('| .:').tokenize
|
||||||
assert_equal [[:pipe, '|'], [:dot, '.'], [:colon, ':'], [:end_of_string]], tokens
|
assert_equal([[:pipe, '|'], [:dot, '.'], [:colon, ':'], [:end_of_string]], tokens)
|
||||||
tokens = Lexer.new('[,]').tokenize
|
tokens = Lexer.new('[,]').tokenize
|
||||||
assert_equal [[:open_square, '['], [:comma, ','], [:close_square, ']'], [:end_of_string]], tokens
|
assert_equal([[:open_square, '['], [:comma, ','], [:close_square, ']'], [:end_of_string]], tokens)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_fancy_identifiers
|
def test_fancy_identifiers
|
||||||
tokens = Lexer.new('hi five?').tokenize
|
tokens = Lexer.new('hi five?').tokenize
|
||||||
assert_equal [[:id, 'hi'], [:id, 'five?'], [:end_of_string]], tokens
|
assert_equal([[:id, 'hi'], [:id, 'five?'], [:end_of_string]], tokens)
|
||||||
|
|
||||||
tokens = Lexer.new('2foo').tokenize
|
tokens = Lexer.new('2foo').tokenize
|
||||||
assert_equal [[:number, '2'], [:id, 'foo'], [:end_of_string]], tokens
|
assert_equal([[:number, '2'], [:id, 'foo'], [:end_of_string]], tokens)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_whitespace
|
def test_whitespace
|
||||||
tokens = Lexer.new("five|\n\t ==").tokenize
|
tokens = Lexer.new("five|\n\t ==").tokenize
|
||||||
assert_equal [[:id, 'five'], [:pipe, '|'], [:comparison, '=='], [:end_of_string]], tokens
|
assert_equal([[:id, 'five'], [:pipe, '|'], [:comparison, '=='], [:end_of_string]], tokens)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_unexpected_character
|
def test_unexpected_character
|
||||||
|
|||||||
@@ -238,7 +238,7 @@ class ParseTreeVisitorTest < Minitest::Test
|
|||||||
def traversal(template)
|
def traversal(template)
|
||||||
ParseTreeVisitor
|
ParseTreeVisitor
|
||||||
.for(Template.parse(template).root)
|
.for(Template.parse(template).root)
|
||||||
.add_callback_for(VariableLookup, &:name)
|
.add_callback_for(VariableLookup) { |node| node.name } # rubocop:disable Style/SymbolProc
|
||||||
end
|
end
|
||||||
|
|
||||||
def visit(template)
|
def visit(template)
|
||||||
@@ -7,72 +7,72 @@ class ParserUnitTest < Minitest::Test
|
|||||||
|
|
||||||
def test_consume
|
def test_consume
|
||||||
p = Parser.new("wat: 7")
|
p = Parser.new("wat: 7")
|
||||||
assert_equal 'wat', p.consume(:id)
|
assert_equal('wat', p.consume(:id))
|
||||||
assert_equal ':', p.consume(:colon)
|
assert_equal(':', p.consume(:colon))
|
||||||
assert_equal '7', p.consume(:number)
|
assert_equal('7', p.consume(:number))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_jump
|
def test_jump
|
||||||
p = Parser.new("wat: 7")
|
p = Parser.new("wat: 7")
|
||||||
p.jump(2)
|
p.jump(2)
|
||||||
assert_equal '7', p.consume(:number)
|
assert_equal('7', p.consume(:number))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_consume?
|
def test_consume?
|
||||||
p = Parser.new("wat: 7")
|
p = Parser.new("wat: 7")
|
||||||
assert_equal 'wat', p.consume?(:id)
|
assert_equal('wat', p.consume?(:id))
|
||||||
assert_equal false, p.consume?(:dot)
|
assert_equal(false, p.consume?(:dot))
|
||||||
assert_equal ':', p.consume(:colon)
|
assert_equal(':', p.consume(:colon))
|
||||||
assert_equal '7', p.consume?(:number)
|
assert_equal('7', p.consume?(:number))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_id?
|
def test_id?
|
||||||
p = Parser.new("wat 6 Peter Hegemon")
|
p = Parser.new("wat 6 Peter Hegemon")
|
||||||
assert_equal 'wat', p.id?('wat')
|
assert_equal('wat', p.id?('wat'))
|
||||||
assert_equal false, p.id?('endgame')
|
assert_equal(false, p.id?('endgame'))
|
||||||
assert_equal '6', p.consume(:number)
|
assert_equal('6', p.consume(:number))
|
||||||
assert_equal 'Peter', p.id?('Peter')
|
assert_equal('Peter', p.id?('Peter'))
|
||||||
assert_equal false, p.id?('Achilles')
|
assert_equal(false, p.id?('Achilles'))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_look
|
def test_look
|
||||||
p = Parser.new("wat 6 Peter Hegemon")
|
p = Parser.new("wat 6 Peter Hegemon")
|
||||||
assert_equal true, p.look(:id)
|
assert_equal(true, p.look(:id))
|
||||||
assert_equal 'wat', p.consume(:id)
|
assert_equal('wat', p.consume(:id))
|
||||||
assert_equal false, p.look(:comparison)
|
assert_equal(false, p.look(:comparison))
|
||||||
assert_equal true, p.look(:number)
|
assert_equal(true, p.look(:number))
|
||||||
assert_equal true, p.look(:id, 1)
|
assert_equal(true, p.look(:id, 1))
|
||||||
assert_equal false, p.look(:number, 1)
|
assert_equal(false, p.look(:number, 1))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_expressions
|
def test_expressions
|
||||||
p = Parser.new("hi.there hi?[5].there? hi.there.bob")
|
p = Parser.new("hi.there hi?[5].there? hi.there.bob")
|
||||||
assert_equal 'hi.there', p.expression
|
assert_equal('hi.there', p.expression)
|
||||||
assert_equal 'hi?[5].there?', p.expression
|
assert_equal('hi?[5].there?', p.expression)
|
||||||
assert_equal 'hi.there.bob', p.expression
|
assert_equal('hi.there.bob', p.expression)
|
||||||
|
|
||||||
p = Parser.new("567 6.0 'lol' \"wut\"")
|
p = Parser.new("567 6.0 'lol' \"wut\"")
|
||||||
assert_equal '567', p.expression
|
assert_equal('567', p.expression)
|
||||||
assert_equal '6.0', p.expression
|
assert_equal('6.0', p.expression)
|
||||||
assert_equal "'lol'", p.expression
|
assert_equal("'lol'", p.expression)
|
||||||
assert_equal '"wut"', p.expression
|
assert_equal('"wut"', p.expression)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_ranges
|
def test_ranges
|
||||||
p = Parser.new("(5..7) (1.5..9.6) (young..old) (hi[5].wat..old)")
|
p = Parser.new("(5..7) (1.5..9.6) (young..old) (hi[5].wat..old)")
|
||||||
assert_equal '(5..7)', p.expression
|
assert_equal('(5..7)', p.expression)
|
||||||
assert_equal '(1.5..9.6)', p.expression
|
assert_equal('(1.5..9.6)', p.expression)
|
||||||
assert_equal '(young..old)', p.expression
|
assert_equal('(young..old)', p.expression)
|
||||||
assert_equal '(hi[5].wat..old)', p.expression
|
assert_equal('(hi[5].wat..old)', p.expression)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_arguments
|
def test_arguments
|
||||||
p = Parser.new("filter: hi.there[5], keyarg: 7")
|
p = Parser.new("filter: hi.there[5], keyarg: 7")
|
||||||
assert_equal 'filter', p.consume(:id)
|
assert_equal('filter', p.consume(:id))
|
||||||
assert_equal ':', p.consume(:colon)
|
assert_equal(':', p.consume(:colon))
|
||||||
assert_equal 'hi.there[5]', p.argument
|
assert_equal('hi.there[5]', p.argument)
|
||||||
assert_equal ',', p.consume(:comma)
|
assert_equal(',', p.consume(:comma))
|
||||||
assert_equal 'keyarg: 7', p.argument
|
assert_equal('keyarg: 7', p.argument)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_invalid_expression
|
def test_invalid_expression
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ class PartialCacheUnitTest < Minitest::Test
|
|||||||
parse_context: Liquid::ParseContext.new
|
parse_context: Liquid::ParseContext.new
|
||||||
)
|
)
|
||||||
|
|
||||||
assert_equal 'my partial body', partial.render
|
assert_equal('my partial body', partial.render)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_reads_from_the_file_system_only_once_per_file
|
def test_reads_from_the_file_system_only_once_per_file
|
||||||
@@ -33,7 +33,7 @@ class PartialCacheUnitTest < Minitest::Test
|
|||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_equal 1, file_system.file_read_count
|
assert_equal(1, file_system.file_read_count)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_cache_state_is_stored_per_context
|
def test_cache_state_is_stored_per_context
|
||||||
@@ -66,7 +66,7 @@ class PartialCacheUnitTest < Minitest::Test
|
|||||||
parse_context: parse_context
|
parse_context: parse_context
|
||||||
)
|
)
|
||||||
|
|
||||||
assert_equal 2, shared_file_system.file_read_count
|
assert_equal(2, shared_file_system.file_read_count)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_cache_is_not_broken_when_a_different_parse_context_is_used
|
def test_cache_is_not_broken_when_a_different_parse_context_is_used
|
||||||
@@ -88,6 +88,41 @@ class PartialCacheUnitTest < Minitest::Test
|
|||||||
|
|
||||||
# Technically what we care about is that the file was parsed twice,
|
# Technically what we care about is that the file was parsed twice,
|
||||||
# but measuring file reads is an OK proxy for this.
|
# but measuring file reads is an OK proxy for this.
|
||||||
assert_equal 1, file_system.file_read_count
|
assert_equal(1, file_system.file_read_count)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_uses_default_template_factory_when_no_template_factory_found_in_register
|
||||||
|
context = Liquid::Context.build(
|
||||||
|
registers: {
|
||||||
|
file_system: StubFileSystem.new('my_partial' => 'my partial body'),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
partial = Liquid::PartialCache.load(
|
||||||
|
'my_partial',
|
||||||
|
context: context,
|
||||||
|
parse_context: Liquid::ParseContext.new
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_equal('my partial body', partial.render)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_uses_template_factory_register_if_present
|
||||||
|
template_factory = StubTemplateFactory.new
|
||||||
|
context = Liquid::Context.build(
|
||||||
|
registers: {
|
||||||
|
file_system: StubFileSystem.new('my_partial' => 'my partial body'),
|
||||||
|
template_factory: template_factory,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
partial = Liquid::PartialCache.load(
|
||||||
|
'my_partial',
|
||||||
|
context: context,
|
||||||
|
parse_context: Liquid::ParseContext.new
|
||||||
|
)
|
||||||
|
|
||||||
|
assert_equal('my partial body', partial.render)
|
||||||
|
assert_equal(1, template_factory.count)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -6,41 +6,41 @@ class RegexpUnitTest < Minitest::Test
|
|||||||
include Liquid
|
include Liquid
|
||||||
|
|
||||||
def test_empty
|
def test_empty
|
||||||
assert_equal [], ''.scan(QuotedFragment)
|
assert_equal([], ''.scan(QuotedFragment))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_quote
|
def test_quote
|
||||||
assert_equal ['"arg 1"'], '"arg 1"'.scan(QuotedFragment)
|
assert_equal(['"arg 1"'], '"arg 1"'.scan(QuotedFragment))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_words
|
def test_words
|
||||||
assert_equal ['arg1', 'arg2'], 'arg1 arg2'.scan(QuotedFragment)
|
assert_equal(['arg1', 'arg2'], 'arg1 arg2'.scan(QuotedFragment))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_tags
|
def test_tags
|
||||||
assert_equal ['<tr>', '</tr>'], '<tr> </tr>'.scan(QuotedFragment)
|
assert_equal(['<tr>', '</tr>'], '<tr> </tr>'.scan(QuotedFragment))
|
||||||
assert_equal ['<tr></tr>'], '<tr></tr>'.scan(QuotedFragment)
|
assert_equal(['<tr></tr>'], '<tr></tr>'.scan(QuotedFragment))
|
||||||
assert_equal ['<style', 'class="hello">', '</style>'], %(<style class="hello">' </style>).scan(QuotedFragment)
|
assert_equal(['<style', 'class="hello">', '</style>'], %(<style class="hello">' </style>).scan(QuotedFragment))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_double_quoted_words
|
def test_double_quoted_words
|
||||||
assert_equal ['arg1', 'arg2', '"arg 3"'], 'arg1 arg2 "arg 3"'.scan(QuotedFragment)
|
assert_equal(['arg1', 'arg2', '"arg 3"'], 'arg1 arg2 "arg 3"'.scan(QuotedFragment))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_single_quoted_words
|
def test_single_quoted_words
|
||||||
assert_equal ['arg1', 'arg2', "'arg 3'"], 'arg1 arg2 \'arg 3\''.scan(QuotedFragment)
|
assert_equal(['arg1', 'arg2', "'arg 3'"], 'arg1 arg2 \'arg 3\''.scan(QuotedFragment))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_quoted_words_in_the_middle
|
def test_quoted_words_in_the_middle
|
||||||
assert_equal ['arg1', 'arg2', '"arg 3"', 'arg4'], 'arg1 arg2 "arg 3" arg4 '.scan(QuotedFragment)
|
assert_equal(['arg1', 'arg2', '"arg 3"', 'arg4'], 'arg1 arg2 "arg 3" arg4 '.scan(QuotedFragment))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_variable_parser
|
def test_variable_parser
|
||||||
assert_equal ['var'], 'var'.scan(VariableParser)
|
assert_equal(['var'], 'var'.scan(VariableParser))
|
||||||
assert_equal ['var', 'method'], 'var.method'.scan(VariableParser)
|
assert_equal(['var', 'method'], 'var.method'.scan(VariableParser))
|
||||||
assert_equal ['var', '[method]'], 'var[method]'.scan(VariableParser)
|
assert_equal(['var', '[method]'], 'var[method]'.scan(VariableParser))
|
||||||
assert_equal ['var', '[method]', '[0]'], 'var[method][0]'.scan(VariableParser)
|
assert_equal(['var', '[method]', '[0]'], 'var[method][0]'.scan(VariableParser))
|
||||||
assert_equal ['var', '["method"]', '[0]'], 'var["method"][0]'.scan(VariableParser)
|
assert_equal(['var', '["method"]', '[0]'], 'var["method"][0]'.scan(VariableParser))
|
||||||
assert_equal ['var', '[method]', '[0]', 'method'], 'var[method][0].method'.scan(VariableParser)
|
assert_equal(['var', '[method]', '[0]', 'method'], 'var[method][0].method'.scan(VariableParser))
|
||||||
end
|
end
|
||||||
end # RegexpTest
|
end # RegexpTest
|
||||||
|
|||||||
@@ -1,36 +0,0 @@
|
|||||||
# frozen_string_literal: true
|
|
||||||
|
|
||||||
require 'test_helper'
|
|
||||||
|
|
||||||
class DisabledTagsUnitTest < Minitest::Test
|
|
||||||
include Liquid
|
|
||||||
|
|
||||||
def test_disables_tag_specified
|
|
||||||
register = DisabledTags.new
|
|
||||||
register.disable(%w(foo bar)) do
|
|
||||||
assert_equal true, register.disabled?("foo")
|
|
||||||
assert_equal true, register.disabled?("bar")
|
|
||||||
assert_equal false, register.disabled?("unknown")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_disables_nested_tags
|
|
||||||
register = DisabledTags.new
|
|
||||||
register.disable(["foo"]) do
|
|
||||||
register.disable(["foo"]) do
|
|
||||||
assert_equal true, register.disabled?("foo")
|
|
||||||
assert_equal false, register.disabled?("bar")
|
|
||||||
end
|
|
||||||
register.disable(["bar"]) do
|
|
||||||
assert_equal true, register.disabled?("foo")
|
|
||||||
assert_equal true, register.disabled?("bar")
|
|
||||||
register.disable(["foo"]) do
|
|
||||||
assert_equal true, register.disabled?("foo")
|
|
||||||
assert_equal true, register.disabled?("bar")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
assert_equal true, register.disabled?("foo")
|
|
||||||
assert_equal false, register.disabled?("bar")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -5,244 +5,152 @@ require 'test_helper'
|
|||||||
class StaticRegistersUnitTest < Minitest::Test
|
class StaticRegistersUnitTest < Minitest::Test
|
||||||
include Liquid
|
include Liquid
|
||||||
|
|
||||||
def set
|
def test_set
|
||||||
static_register = StaticRegisters.new
|
static_register = StaticRegisters.new(a: 1, b: 2)
|
||||||
static_register[nil] = true
|
static_register[:b] = 22
|
||||||
static_register[1] = :one
|
static_register[:c] = 33
|
||||||
static_register[:one] = "one"
|
|
||||||
static_register["two"] = "three"
|
|
||||||
static_register["two"] = 3
|
|
||||||
static_register[false] = nil
|
|
||||||
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, static_register.registers)
|
assert_equal(1, static_register[:a])
|
||||||
|
assert_equal(22, static_register[:b])
|
||||||
static_register
|
assert_equal(33, static_register[:c])
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_get
|
def test_get_missing_key
|
||||||
static_register = set
|
static_register = StaticRegisters.new
|
||||||
|
|
||||||
assert_equal true, static_register[nil]
|
assert_nil(static_register[:missing])
|
||||||
assert_equal :one, static_register[1]
|
|
||||||
assert_equal "one", static_register[:one]
|
|
||||||
assert_equal 3, static_register["two"]
|
|
||||||
assert_nil static_register[false]
|
|
||||||
assert_nil static_register["unknown"]
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_delete
|
def test_delete
|
||||||
static_register = set
|
static_register = StaticRegisters.new(a: 1, b: 2)
|
||||||
|
static_register[:b] = 22
|
||||||
|
static_register[:c] = 33
|
||||||
|
|
||||||
assert_equal true, static_register.delete(nil)
|
assert_nil(static_register.delete(:a))
|
||||||
assert_equal :one, static_register.delete(1)
|
|
||||||
assert_equal "one", static_register.delete(:one)
|
|
||||||
assert_equal 3, static_register.delete("two")
|
|
||||||
assert_nil static_register.delete(false)
|
|
||||||
assert_nil static_register.delete("unknown")
|
|
||||||
|
|
||||||
assert_equal({}, static_register.registers)
|
assert_equal(22, static_register.delete(:b))
|
||||||
|
|
||||||
|
assert_equal(33, static_register.delete(:c))
|
||||||
|
assert_nil(static_register[:c])
|
||||||
|
|
||||||
|
assert_nil(static_register.delete(:d))
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_fetch
|
def test_fetch
|
||||||
static_register = set
|
static_register = StaticRegisters.new(a: 1, b: 2)
|
||||||
|
static_register[:b] = 22
|
||||||
|
static_register[:c] = 33
|
||||||
|
|
||||||
assert_equal true, static_register.fetch(nil)
|
assert_equal(1, static_register.fetch(:a))
|
||||||
assert_equal :one, static_register.fetch(1)
|
assert_equal(1, static_register.fetch(:a, "default"))
|
||||||
assert_equal "one", static_register.fetch(:one)
|
assert_equal(22, static_register.fetch(:b))
|
||||||
assert_equal 3, static_register.fetch("two")
|
assert_equal(22, static_register.fetch(:b, "default"))
|
||||||
assert_nil static_register.fetch(false)
|
assert_equal(33, static_register.fetch(:c))
|
||||||
assert_nil static_register.fetch("unknown")
|
assert_equal(33, static_register.fetch(:c, "default"))
|
||||||
|
|
||||||
|
assert_raises(KeyError) do
|
||||||
|
static_register.fetch(:d)
|
||||||
end
|
end
|
||||||
|
assert_equal("default", static_register.fetch(:d, "default"))
|
||||||
|
|
||||||
def test_fetch_default
|
result = static_register.fetch(:d) { "default" }
|
||||||
static_register = StaticRegisters.new
|
assert_equal("default", result)
|
||||||
|
|
||||||
assert_equal true, static_register.fetch(nil, true)
|
result = static_register.fetch(:d, "default 1") { "default 2" }
|
||||||
assert_equal :one, static_register.fetch(1, :one)
|
assert_equal("default 2", result)
|
||||||
assert_equal "one", static_register.fetch(:one, "one")
|
|
||||||
assert_equal 3, static_register.fetch("two", 3)
|
|
||||||
assert_nil static_register.fetch(false, nil)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_key
|
def test_key
|
||||||
static_register = set
|
static_register = StaticRegisters.new(a: 1, b: 2)
|
||||||
|
static_register[:b] = 22
|
||||||
|
static_register[:c] = 33
|
||||||
|
|
||||||
assert_equal true, static_register.key?(nil)
|
assert_equal(true, static_register.key?(:a))
|
||||||
assert_equal true, static_register.key?(1)
|
assert_equal(true, static_register.key?(:b))
|
||||||
assert_equal true, static_register.key?(:one)
|
assert_equal(true, static_register.key?(:c))
|
||||||
assert_equal true, static_register.key?("two")
|
assert_equal(false, static_register.key?(:d))
|
||||||
assert_equal true, static_register.key?(false)
|
|
||||||
assert_equal false, static_register.key?("unknown")
|
|
||||||
assert_equal false, static_register.key?(true)
|
|
||||||
end
|
|
||||||
|
|
||||||
def set_with_static
|
|
||||||
static_register = StaticRegisters.new(nil => true, 1 => :one, :one => "one", "two" => 3, false => nil)
|
|
||||||
static_register[nil] = false
|
|
||||||
static_register["two"] = 4
|
|
||||||
static_register[true] = "foo"
|
|
||||||
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, static_register.static)
|
|
||||||
assert_equal({ nil => false, "two" => 4, true => "foo" }, static_register.registers)
|
|
||||||
|
|
||||||
static_register
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_get_with_static
|
|
||||||
static_register = set_with_static
|
|
||||||
|
|
||||||
assert_equal false, static_register[nil]
|
|
||||||
assert_equal :one, static_register[1]
|
|
||||||
assert_equal "one", static_register[:one]
|
|
||||||
assert_equal 4, static_register["two"]
|
|
||||||
assert_equal "foo", static_register[true]
|
|
||||||
assert_nil static_register[false]
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_delete_with_static
|
|
||||||
static_register = set_with_static
|
|
||||||
|
|
||||||
assert_equal false, static_register.delete(nil)
|
|
||||||
assert_equal 4, static_register.delete("two")
|
|
||||||
assert_equal "foo", static_register.delete(true)
|
|
||||||
assert_nil static_register.delete("unknown")
|
|
||||||
assert_nil static_register.delete(:one)
|
|
||||||
|
|
||||||
assert_equal({}, static_register.registers)
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, static_register.static)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_fetch_with_static
|
|
||||||
static_register = set_with_static
|
|
||||||
|
|
||||||
assert_equal false, static_register.fetch(nil)
|
|
||||||
assert_equal :one, static_register.fetch(1)
|
|
||||||
assert_equal "one", static_register.fetch(:one)
|
|
||||||
assert_equal 4, static_register.fetch("two")
|
|
||||||
assert_equal "foo", static_register.fetch(true)
|
|
||||||
assert_nil static_register.fetch(false)
|
|
||||||
end
|
|
||||||
|
|
||||||
def test_key_with_static
|
|
||||||
static_register = set_with_static
|
|
||||||
|
|
||||||
assert_equal true, static_register.key?(nil)
|
|
||||||
assert_equal true, static_register.key?(1)
|
|
||||||
assert_equal true, static_register.key?(:one)
|
|
||||||
assert_equal true, static_register.key?("two")
|
|
||||||
assert_equal true, static_register.key?(false)
|
|
||||||
assert_equal false, static_register.key?("unknown")
|
|
||||||
assert_equal true, static_register.key?(true)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_static_register_can_be_frozen
|
def test_static_register_can_be_frozen
|
||||||
static_register = set_with_static
|
static_register = StaticRegisters.new(a: 1)
|
||||||
|
|
||||||
static = static_register.static.freeze
|
static_register.static.freeze
|
||||||
|
|
||||||
assert_raises(RuntimeError) do
|
assert_raises(RuntimeError) do
|
||||||
static["two"] = "foo"
|
static_register.static[:a] = "foo"
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_raises(RuntimeError) do
|
assert_raises(RuntimeError) do
|
||||||
static["unknown"] = "foo"
|
static_register.static[:b] = "foo"
|
||||||
end
|
end
|
||||||
|
|
||||||
assert_raises(RuntimeError) do
|
assert_raises(RuntimeError) do
|
||||||
static.delete("two")
|
static_register.static.delete(:a)
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_raises(RuntimeError) do
|
||||||
|
static_register.static.delete(:c)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_new_static_retains_static
|
def test_new_static_retains_static
|
||||||
static_register = StaticRegisters.new(nil => true, 1 => :one, :one => "one", "two" => 3, false => nil)
|
static_register = StaticRegisters.new(a: 1, b: 2)
|
||||||
static_register["one"] = 1
|
static_register[:b] = 22
|
||||||
static_register["two"] = 2
|
static_register[:c] = 33
|
||||||
static_register["three"] = 3
|
|
||||||
|
|
||||||
new_register = StaticRegisters.new(static_register)
|
new_static_register = StaticRegisters.new(static_register)
|
||||||
assert_equal({}, new_register.registers)
|
new_static_register[:b] = 222
|
||||||
|
|
||||||
new_register["one"] = 4
|
newest_static_register = StaticRegisters.new(new_static_register)
|
||||||
new_register["two"] = 5
|
newest_static_register[:c] = 333
|
||||||
new_register["three"] = 6
|
|
||||||
|
|
||||||
newest_register = StaticRegisters.new(new_register)
|
assert_equal(1, static_register[:a])
|
||||||
assert_equal({}, newest_register.registers)
|
assert_equal(22, static_register[:b])
|
||||||
|
assert_equal(33, static_register[:c])
|
||||||
|
|
||||||
newest_register["one"] = 7
|
assert_equal(1, new_static_register[:a])
|
||||||
newest_register["two"] = 8
|
assert_equal(222, new_static_register[:b])
|
||||||
newest_register["three"] = 9
|
assert_nil(new_static_register[:c])
|
||||||
|
|
||||||
assert_equal({ "one" => 1, "two" => 2, "three" => 3 }, static_register.registers)
|
assert_equal(1, newest_static_register[:a])
|
||||||
assert_equal({ "one" => 4, "two" => 5, "three" => 6 }, new_register.registers)
|
assert_equal(2, newest_static_register[:b])
|
||||||
assert_equal({ "one" => 7, "two" => 8, "three" => 9 }, newest_register.registers)
|
assert_equal(333, newest_static_register[:c])
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, static_register.static)
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, new_register.static)
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, newest_register.static)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_multiple_instances_are_unique
|
def test_multiple_instances_are_unique
|
||||||
static_register = StaticRegisters.new(nil => true, 1 => :one, :one => "one", "two" => 3, false => nil)
|
static_register_1 = StaticRegisters.new(a: 1, b: 2)
|
||||||
static_register["one"] = 1
|
static_register_1[:b] = 22
|
||||||
static_register["two"] = 2
|
static_register_1[:c] = 33
|
||||||
static_register["three"] = 3
|
|
||||||
|
|
||||||
new_register = StaticRegisters.new(foo: :bar)
|
static_register_2 = StaticRegisters.new(a: 10, b: 20)
|
||||||
assert_equal({}, new_register.registers)
|
static_register_2[:b] = 220
|
||||||
|
static_register_2[:c] = 330
|
||||||
|
|
||||||
new_register["one"] = 4
|
assert_equal({ a: 1, b: 2 }, static_register_1.static)
|
||||||
new_register["two"] = 5
|
assert_equal(1, static_register_1[:a])
|
||||||
new_register["three"] = 6
|
assert_equal(22, static_register_1[:b])
|
||||||
|
assert_equal(33, static_register_1[:c])
|
||||||
|
|
||||||
newest_register = StaticRegisters.new(bar: :foo)
|
assert_equal({ a: 10, b: 20 }, static_register_2.static)
|
||||||
assert_equal({}, newest_register.registers)
|
assert_equal(10, static_register_2[:a])
|
||||||
|
assert_equal(220, static_register_2[:b])
|
||||||
newest_register["one"] = 7
|
assert_equal(330, static_register_2[:c])
|
||||||
newest_register["two"] = 8
|
|
||||||
newest_register["three"] = 9
|
|
||||||
|
|
||||||
assert_equal({ "one" => 1, "two" => 2, "three" => 3 }, static_register.registers)
|
|
||||||
assert_equal({ "one" => 4, "two" => 5, "three" => 6 }, new_register.registers)
|
|
||||||
assert_equal({ "one" => 7, "two" => 8, "three" => 9 }, newest_register.registers)
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, static_register.static)
|
|
||||||
assert_equal({ foo: :bar }, new_register.static)
|
|
||||||
assert_equal({ bar: :foo }, newest_register.static)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_can_update_static_directly_and_updates_all_instances
|
def test_initialization_reused_static_same_memory_object
|
||||||
static_register = StaticRegisters.new(nil => true, 1 => :one, :one => "one", "two" => 3, false => nil)
|
static_register_1 = StaticRegisters.new(a: 1, b: 2)
|
||||||
static_register["one"] = 1
|
static_register_1[:b] = 22
|
||||||
static_register["two"] = 2
|
static_register_1[:c] = 33
|
||||||
static_register["three"] = 3
|
|
||||||
|
|
||||||
new_register = StaticRegisters.new(static_register)
|
static_register_2 = StaticRegisters.new(static_register_1)
|
||||||
assert_equal({}, new_register.registers)
|
|
||||||
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil }, static_register.static)
|
assert_equal(1, static_register_2[:a])
|
||||||
|
assert_equal(2, static_register_2[:b])
|
||||||
|
assert_nil(static_register_2[:c])
|
||||||
|
|
||||||
new_register["one"] = 4
|
static_register_1.static[:b] = 222
|
||||||
new_register["two"] = 5
|
static_register_1.static[:c] = 333
|
||||||
new_register["three"] = 6
|
|
||||||
new_register.static["four"] = 10
|
|
||||||
|
|
||||||
newest_register = StaticRegisters.new(new_register)
|
assert_same(static_register_1.static, static_register_2.static)
|
||||||
assert_equal({}, newest_register.registers)
|
|
||||||
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil, "four" => 10 }, new_register.static)
|
|
||||||
|
|
||||||
newest_register["one"] = 7
|
|
||||||
newest_register["two"] = 8
|
|
||||||
newest_register["three"] = 9
|
|
||||||
new_register.static["four"] = 5
|
|
||||||
new_register.static["five"] = 15
|
|
||||||
|
|
||||||
assert_equal({ "one" => 1, "two" => 2, "three" => 3 }, static_register.registers)
|
|
||||||
assert_equal({ "one" => 4, "two" => 5, "three" => 6 }, new_register.registers)
|
|
||||||
assert_equal({ "one" => 7, "two" => 8, "three" => 9 }, newest_register.registers)
|
|
||||||
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil, "four" => 5, "five" => 15 }, newest_register.static)
|
|
||||||
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil, "four" => 5, "five" => 15 }, static_register.static)
|
|
||||||
assert_equal({ nil => true, 1 => :one, :one => "one", "two" => 3, false => nil, "four" => 5, "five" => 15 }, new_register.static)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user