Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

recursively parse brackets on variable lookup #1680

Merged
merged 2 commits into from
Jan 31, 2023

Conversation

ggmichaelgo
Copy link
Contributor

@ggmichaelgo ggmichaelgo commented Jan 30, 2023

Problem

With dynamic variable lookup, Liquid is not properly parsing the expression.
With this template:

{{ [list[settings.zero]] }}

VariableLookup matches [list[settings.zero] for the name without the extra closing bracket.
This causes an issue because VariableLookup parses the expression with Expression.parse(name[1..-2]) which is list[settings.zero and Liquid creates a VariableLookup where the name is list and lookups are ["settings", "zero"].

This problem also exists for the double-nested variable lookups such as this template: list[llist[setting.value]]

Solution

We can solve this issue by recursively parsing the dynamic lookups:

\[(?:[^\[\]]+|\g<0>)*\]

\[          # match open bracket
(?:         # start non-capturing group
  [^\[\]]+  # match non-square brackets
  |         # OR
  \g<0>     # go back to the beginning of the regex
)*          # end of non-capturing group
\]          # match closing bracket

@ggmichaelgo ggmichaelgo force-pushed the recursively-parse-brackets branch 4 times, most recently from 3da2014 to eee6c31 Compare January 31, 2023 00:13
@ggmichaelgo ggmichaelgo marked this pull request as ready for review January 31, 2023 00:16
@ggmichaelgo ggmichaelgo force-pushed the recursively-parse-brackets branch from eee6c31 to 1aaf6ed Compare January 31, 2023 00:19
@dylanahsmith
Copy link
Contributor

Recursive parsing is actually something that Liquid::Parser does correctly, which would cause it to recognize this syntax as strictly valid. However, it then builds an expression String that it passes to the lax parser, which causes it to have this behaviour.

liquid-c actually parses this properly when the disable_liquid_c_nodes: true parse option isn't being used, since it will build the expression directly when the liquid code is strictly parseable.

As such, it seems like this is more clearly a bug in the strict parser than the lax parser.

This causes an issue because VariableLookup parses the expression with Expression.parse(name[1..-2]) which is list[settings.zero and Liquid creates a VariableLookup where the name is list and lookups are ["settings", "zero"].

Unfortunately, this highlights how hard it is to tell whether a liquid change will break existing code, since this can lead to code that accidentally works or fails more gracefully (e.g. avoiding broken conditional code). That means that if it is too disruptive, we may need to revert, but we would still want the fix to the strict parser.

Fundamentally, the approach taken to the strict parser is quite fragile and inefficient by parsing, rebuilding the markup, then using the lax parser to parse that rebuilt markup. Although, it looks like a fairly large refactor might be needed to support that (e.g. constructors are coupled to parsing and there is some quirky coupling to markup). So it would be nice to not depend on that work to fix this liquid-c inconsistency.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants