Skip to content

Commit

Permalink
Merge pull request #9 from jg-rp/template-strings
Browse files Browse the repository at this point in the history
Implement template strings
  • Loading branch information
jg-rp authored Dec 30, 2024
2 parents f78d26b + 374ab0f commit e81f022
Show file tree
Hide file tree
Showing 13 changed files with 575 additions and 50 deletions.
3 changes: 2 additions & 1 deletion docs/migration.md
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ Whether shopify/Liquid compatibility is important to you or not, if you’re dev
The following features are new or are now built-in where they weren't before.

- More whitespace control. Along with a `default_trim` configuration option, tags and the output statement now support `+`, `-` and `~` for controlling whitespace in templates. By default, `~` will remove newlines but retain space and tab characters.
- String literals support interpolation using `${` and `}` as delimiters. For example, `{% echo 'Hello, ${you | capitalize}' %}`.
- Logical expressions now support negation with the `not` operator and grouping terms with parentheses by default.
- Ternary expressions are now available by default. For example, `{{ a if b else c }}` or `{{ a | upcase if b == 'foo' else c || split }}`.
- Inline comments surrounded by `{#` and `#}` are enabled by default. Additional `#`’s can be added to comment out blocks of markup that already contain comments, as long as the number of hashes match.
Expand Down Expand Up @@ -83,7 +84,7 @@ TODO:

The following packages are dependencies of Python Liquid2.

- Markupsafe>=2
- Markupsafe>=3
- Babel>=2
- python-dateutil
- pytz
2 changes: 2 additions & 0 deletions liquid2/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
from .token import is_raw_token
from .token import is_tag_token
from .token import is_token_type
from .token import is_template_string_token
from .stream import TokenStream
from .expression import Expression
from .tag import Tag
Expand Down Expand Up @@ -167,6 +168,7 @@ def extract_liquid(
"is_range_token",
"is_raw_token",
"is_tag_token",
"is_template_string_token",
"is_token_type",
"LinesToken",
"Node",
Expand Down
118 changes: 115 additions & 3 deletions liquid2/builtin/expressions.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,13 +19,17 @@
from typing import cast

from markupsafe import Markup
from markupsafe import escape

from liquid2 import PathToken
from liquid2 import RenderContext
from liquid2 import Token
from liquid2 import TokenStream
from liquid2 import TokenType
from liquid2 import is_output_token
from liquid2 import is_path_token
from liquid2 import is_range_token
from liquid2 import is_template_string_token
from liquid2 import is_token_type
from liquid2.exceptions import LiquidSyntaxError
from liquid2.exceptions import LiquidTypeError
Expand All @@ -34,9 +38,9 @@
from liquid2.unescape import unescape

if TYPE_CHECKING:
from liquid2 import OutputToken
from liquid2 import PathT
from liquid2 import RenderContext
from liquid2 import TokenStream
from liquid2 import TokenT


Expand Down Expand Up @@ -300,6 +304,67 @@ def children(self) -> list[Expression]:
return [self.start, self.stop]


class TemplateString(Expression):
__slots__ = ("template",)

def __init__(self, token: TokenT, template: list[Token | OutputToken]):
super().__init__(token)
self.template: list[Expression] = []

for _token in template:
if is_token_type(_token, TokenType.SINGLE_QUOTE_STRING):
self.template.append(
StringLiteral(
_token, unescape(_token.value.replace("\\'", "'"), token=_token)
)
)
elif is_token_type(_token, TokenType.DOUBLE_QUOTE_STRING):
self.template.append(
StringLiteral(_token, unescape(_token.value, token=_token))
)
elif is_output_token(_token):
self.template.append(
FilteredExpression.parse(TokenStream(_token.expression))
)
else:
raise LiquidSyntaxError(
"unexpected token in template string", token=_token
)

def __eq__(self, other: object) -> bool:
return isinstance(other, TemplateString) and self.template == other.template

def __str__(self) -> str:
return repr(
"".join(
e.value if isinstance(e, StringLiteral) else f"${{{e}}}"
for e in self.template
)
)

def __hash__(self) -> int:
return hash(tuple(self.template))

def __sizeof__(self) -> int:
return sum(sys.getsizeof(expr) for expr in self.template)

def evaluate(self, context: RenderContext) -> str:
return "".join(
_to_liquid_string(expr.evaluate(context)) for expr in self.template
)

async def evaluate_async(self, context: RenderContext) -> object:
return "".join(
[
_to_liquid_string(await expr.evaluate_async(context))
for expr in self.template
]
)

def children(self) -> list[Expression]:
return self.template


RE_PROPERTY = re.compile(r"[\u0080-\uFFFFa-zA-Z_][\u0080-\uFFFFa-zA-Z0-9_-]*")
Segments: TypeAlias = tuple[Union[str, int, "Segments"], ...]

Expand Down Expand Up @@ -475,6 +540,9 @@ def parse_primitive(token: TokenT) -> Expression: # noqa: PLR0911
token, unescape(token.value.replace("\\'", "'"), token=token)
)

if is_template_string_token(token):
return TemplateString(token, token.template)

if is_path_token(token):
return Path(token, token.path)

Expand Down Expand Up @@ -713,6 +781,10 @@ def parse( # noqa: PLR0912
filter_arguments.append(
PositionalArgument(Path(token, [token.value]))
)
elif is_template_string_token(token):
filter_arguments.append(
PositionalArgument(TemplateString(token, token.template))
)
elif is_path_token(token):
filter_arguments.append(
PositionalArgument(Path(token, token.path))
Expand Down Expand Up @@ -914,6 +986,8 @@ def parse_boolean_primitive( # noqa: PLR0912
left = StringLiteral(
token, unescape(token.value.replace("\\'", "'"), token=token)
)
elif is_template_string_token(token):
left = TemplateString(token, token.template)
elif is_path_token(token):
left = Path(token, token.path)
elif is_range_token(token):
Expand Down Expand Up @@ -1584,7 +1658,10 @@ def parse_identifier(token: TokenT) -> Identifier:


def parse_string_or_identifier(token: TokenT) -> Identifier:
"""Parse _token_ as an identifier or a string literal."""
"""Parse _token_ as an identifier or a string literal.
Excludes template strings.
"""
if is_token_type(token, TokenType.DOUBLE_QUOTE_STRING):
return Identifier(unescape(token.value, token=token), token=token)

Expand All @@ -1603,7 +1680,10 @@ def parse_string_or_identifier(token: TokenT) -> Identifier:


def parse_string_or_path(token: TokenT) -> StringLiteral | Path:
"""Parse _token_ as a string literal or a path."""
"""Parse _token_ as a string literal or a path.
Excludes template strings.
"""
if is_token_type(token, TokenType.WORD):
return Path(token, [token.value])

Expand Down Expand Up @@ -1789,3 +1869,35 @@ def _contains(token: TokenT, left: object, right: object) -> bool:
f"and '{right.__class__.__name__}'",
token=token,
)


# XXX: copied to avoid import issues
def _to_liquid_string(val: Any, *, auto_escape: bool = False) -> str:
"""Stringify a Python object ready for output in a Liquid template."""
if isinstance(val, str) or (auto_escape and hasattr(val, "__html__")):
pass
elif isinstance(val, bool):
val = str(val).lower()
elif val is None:
val = ""
elif isinstance(val, range):
val = f"{val.start}..{val.stop - 1}"
elif isinstance(val, Sequence):
if auto_escape:
val = Markup("").join(
_to_liquid_string(itm, auto_escape=auto_escape) for itm in val
)
else:
val = "".join(
_to_liquid_string(itm, auto_escape=auto_escape) for itm in val
)
elif isinstance(val, (Empty, Blank)):
val = ""
else:
val = str(val)

if auto_escape:
val = escape(val)

assert isinstance(val, str)
return val
Loading

0 comments on commit e81f022

Please sign in to comment.