feature #1503 Use LimitIterator on Iterable objects within the slice filter (Anthony...
authorFabien Potencier <fabien.potencier@gmail.com>
Wed, 1 Oct 2014 17:09:15 +0000 (19:09 +0200)
committerFabien Potencier <fabien.potencier@gmail.com>
Wed, 1 Oct 2014 17:09:15 +0000 (19:09 +0200)
commit1ed333bfb9d87ae8e2dfe6bb62b54f494d494042
tree86030d4592ee3a30a5930b86a8ed7cd8384afda4
parent0a179e801a31bd00a0366c1a9ef60a7f9df5243e
parentfdddb1c91e9ae7044edaa74d64deb3c40332a6a5
feature #1503 Use LimitIterator on Iterable objects within the slice filter (Anthony Sterling)

This PR was merged into the 1.16-dev branch.

Discussion
----------

Use LimitIterator on Iterable objects within the slice filter

Previously the `slice` filter would create an array of items if the subject to be sliced implemented Traversable, this PR instead passes the subject to LimitIterator first - but only if `$start` and `$length` are positive values as LimitIterator does not support the "n'th from end" behaviour.

Our use case is that we have an API client, which we pass to slice, that implements Traversable but lazy loads the items to be returned via an API when `Iterator::current()` is called. Here a quick example:-

```
{% for item in client.items | slice(0,5) %}
    {{ item.name }}
{% endfor %}

<img src="/assets/useless-sodding-banner.jpg" />

{% for item in client.items | slice(5,10) %}
    {{ item.name }}
{% endfor %}
```

Current behaviour would mean that the client would issue 15 requests to the API, 5 for the first block, and 10 for the second (but only return items 5 to 10). This PR means only 10 requests are made, 5 in each block.

I'd imagine this would lower memory usage too.

Commits
-------

fdddb1c Use LimitIterator on Iterable objects within the splice filter
lib/Twig/Extension/Core.php