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

feat(corelib): core::iter::zip #7050

Open
wants to merge 8 commits into
base: main
Choose a base branch
from

Conversation

julio4
Copy link
Contributor

@julio4 julio4 commented Jan 11, 2025

'Zips up' two iterators into a single iterator of pairs.

zip() returns a new iterator that will iterate over two other
iterators, returning a tuple where the first element comes from the
first iterator, and the second element comes from the second iterator.

In other words, it zips two iterators together, into a single one.

If either iterator returns Option::None, next from the zipped iterator
will return Option::None.
If the zipped iterator has no more elements to return then each further attempt to advance
it will first try to advance the first iterator at most one time and if it still yielded an
item try to advance the second iterator at most one time.

Examples

Basic usage:

let mut iter = array![1, 2, 3].into_iter().zip(array![4, 5, 6].into_iter());

assert_eq!(iter.next(), Option::Some((1, 4)));
assert_eq!(iter.next(), Option::Some((2, 5)));
assert_eq!(iter.next(), Option::Some((3, 6)));
assert_eq!(iter.next(), Option::None);

@reviewable-StarkWare
Copy link

This change is Reviewable

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewed 3 of 4 files at r1, all commit messages.
Reviewable status: 3 of 4 files reviewed, 2 unresolved discussions (waiting on @julio4)


corelib/src/iter/adapters/zip_adapter.cairo line 70 at r1 (raw file):

    +Destruct<IB>,
> of Iterator<Zip<A, B>> {
    type Item = (IA, IB);

probably preferable:

Suggestion:

impl ZipIterator<
    A,
    B,
    impl IterA: Iterator<A>,
    impl IterB: Iterator<B>,
    +Destruct<A>,
    +Destruct<B>,
    +Destruct<IterA::Item>,
    +Destruct<IterB::Item>,
> of Iterator<Zip<A, B>> {
    type Item = (IterA::Item, IterB::Item);

corelib/src/test/iter_test.cairo line 25 at r1 (raw file):

    assert_eq!(iter.next(), Option::Some(((2, 5), 8)));
    assert_eq!(iter.next(), Option::Some(((3, 6), 9)));
    assert_eq!(iter.next(), Option::None);

Suggestion:

    let mut iter = zip(array![1, 2, 3], array![4, 5, 6]);

    assert_eq!(iter.next(), Option::Some((1, 4)));
    assert_eq!(iter.next(), Option::Some((2, 5)));
    assert_eq!(iter.next(), Option::Some((3, 6)));
    assert_eq!(iter.next(), Option::None);

    // Nested zips are also possible:
    let mut iter = zip(zip(array![1, 2, 3], array![4, 5, 6]), array![7, 8, 9]);

    assert_eq!(iter.next(), Option::Some(((1, 4), 7)));
    assert_eq!(iter.next(), Option::Some(((2, 5), 8)));
    assert_eq!(iter.next(), Option::Some(((3, 6), 9)));
    assert_eq!(iter.next(), Option::None);

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 3 of 4 files reviewed, 3 unresolved discussions (waiting on @julio4)


corelib/src/iter/adapters/zip_adapter.cairo line 36 at r1 (raw file):

/// let zs = array![7, 8, 9];
///
/// let mut iter = zip(zip(xs, ys), zs);

inline arrays - same as in test.

Code quote:

/// let xs = array![1, 2, 3];
/// let ys = array![4, 5, 6];
///
/// let mut iter = zip(xs, ys);
///
/// assert_eq!(iter.next(), Option::Some((1, 4)));
/// assert_eq!(iter.next(), Option::Some((2, 5)));
/// assert_eq!(iter.next(), Option::Some((3, 6)));
/// assert_eq!(iter.next(), Option::None);
///
/// // Nested zips are also possible:
/// let xs = array![1, 2, 3];
/// let ys = array![4, 5, 6];
/// let zs = array![7, 8, 9];
///
/// let mut iter = zip(zip(xs, ys), zs);

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 3 of 4 files reviewed, 3 unresolved discussions (waiting on @julio4)


a discussion (no related file):
@TomerStarkware for 2nd eye

Copy link
Contributor

@enitrat enitrat left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 3 of 4 files reviewed, 4 unresolved discussions (waiting on @julio4 and @orizi)


corelib/src/iter/adapters/zip_adapter.cairo line 15 at r1 (raw file):

/// Converts the arguments to iterators and zips them.
///

You can pre-emptively link this to [Iterator::zip] as I guess that's going to come next

Code snippet:

/// See the documentation of [`Iterator::zip`] for more.

Copy link
Contributor Author

@julio4 julio4 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 3 of 4 files reviewed, 4 unresolved discussions (waiting on @enitrat and @orizi)


corelib/src/iter/adapters/zip_adapter.cairo line 36 at r1 (raw file):

Previously, orizi wrote…

inline arrays - same as in test.

Done.
However I would argue that it's a bit more readable without inlining, especially for nested zip, and it could make sense to make it as readable as possible in the documentation.


corelib/src/iter/adapters/zip_adapter.cairo line 70 at r1 (raw file):

Previously, orizi wrote…

probably preferable:

Done.


corelib/src/test/iter_test.cairo line 25 at r1 (raw file):

    assert_eq!(iter.next(), Option::Some(((2, 5), 8)));
    assert_eq!(iter.next(), Option::Some(((3, 6), 9)));
    assert_eq!(iter.next(), Option::None);

Done.

Copy link
Contributor Author

@julio4 julio4 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 2 of 4 files reviewed, 4 unresolved discussions (waiting on @enitrat and @orizi)


corelib/src/iter/adapters/zip_adapter.cairo line 15 at r1 (raw file):

Previously, enitrat (Mathieu) wrote…

You can pre-emptively link this to [Iterator::zip] as I guess that's going to come next

There's still some issue with default impl so I prefer not blocking thing and I'll add in a separate PR

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

:lgtm:

Reviewed 1 of 2 files at r2, all commit messages.
Reviewable status: 3 of 4 files reviewed, 1 unresolved discussion (waiting on @enitrat)


corelib/src/iter/adapters/zip_adapter.cairo line 36 at r1 (raw file):

Previously, julio4 (Julio) wrote…

Done.
However I would argue that it's a bit more readable without inlining, especially for nested zip, and it could make sense to make it as readable as possible in the documentation.

in the nested - possibly - i don't like it much in cases when there's no actual meaning to the arrays themselves.

Copy link
Contributor Author

@julio4 julio4 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 3 of 4 files reviewed, 1 unresolved discussion (waiting on @enitrat and @orizi)


corelib/src/iter/adapters/zip_adapter.cairo line 36 at r1 (raw file):

Previously, orizi wrote…

in the nested - possibly - i don't like it much in cases when there's no actual meaning to the arrays themselves.

Ok! Should I still keep the arrays for nested case in the documentation or no need?

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewed 1 of 2 files at r2.
Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)

@julio4 julio4 force-pushed the feat/iter_adapter_zip branch 2 times, most recently from 1d12b71 to d2a73a3 Compare January 14, 2025 10:06
Copy link
Contributor Author

@julio4 julio4 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I still have an issue using +IntoIterator<U> in Iterator::zip.
Using +Iterator<U> works fine, but is less flexible.

Reviewable status: 0 of 6 files reviewed, 1 unresolved discussion (waiting on @enitrat and @orizi)

@julio4 julio4 force-pushed the feat/iter_adapter_zip branch from 1941ab9 to ec0571a Compare January 14, 2025 10:09
Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should mostly work now - please rebase and report.

Reviewed 4 of 6 files at r3, 2 of 2 files at r4, all commit messages.
Reviewable status: all files reviewed, 3 unresolved discussions (waiting on @enitrat and @julio4)


corelib/src/iter/adapters/zip_adapter.cairo line 36 at r1 (raw file):

Previously, julio4 (Julio) wrote…

Ok! Should I still keep the arrays for nested case in the documentation or no need?

dealer's choice.


corelib/src/iter/traits/iterator.cairo line 127 at r4 (raw file):

    /// [`next`]: Iterator::next
    #[inline]
    fn zip<U, +Iterator<U> //, +IntoIterator<U>

why not intoiterator?

Code quote:

    fn zip<U, +Iterator<U> //, +IntoIterator<U>

corelib/src/iter/traits/iterator.cairo line 132 at r4 (raw file):

    ) -> Zip<T, U> {
        zipped_iterator(self, other //.into_iter()
        )

Suggestion:

        zipped_iterator(self, other)

Copy link
Contributor Author

@julio4 julio4 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fixed it, see my other comment!
Lastly, would it be possible to add a first simple version of zip! macro that take fixed amount of 2 iterators in argument?
zip!(a: +IntoIterator<A>, b: +IntoIterator<B>) -> a.into_iter().zip(b.into_iter())
It would be possible to extend it to arbitrary arguments count later on without introducing breaking changes.

Reviewable status: 4 of 6 files reviewed, 3 unresolved discussions (waiting on @enitrat and @orizi)


corelib/src/iter/traits/iterator.cairo line 127 at r4 (raw file):

Previously, orizi wrote…

why not intoiterator?

I had an issue using IntoIterator where it always failed with 'Error: Compilation failed without any diagnostics.'
I found that it was missing trait bound +Destruct<T>, which was not reported by the LSP and the compiler.

@julio4 julio4 force-pushed the feat/iter_adapter_zip branch from f3c7152 to ee750a4 Compare January 14, 2025 10:40
Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

hmm - looking at rust now - zip mostly makes sense in any case to be added as a function (unlike chain) - as multi zip! is much more problematic.
:lgtm:

Reviewed 3 of 3 files at r5, all commit messages.
Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)


corelib/src/iter/traits/iterator.cairo line 127 at r4 (raw file):

Previously, julio4 (Julio) wrote…

I had an issue using IntoIterator where it always failed with 'Error: Compilation failed without any diagnostics.'
I found that it was missing trait bound +Destruct<T>, which was not reported by the LSP and the compiler.

the diagnostics should be shown on main now.

Copy link
Contributor

@enitrat enitrat left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

:lgtm:

Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @julio4)

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

:lgtm:

Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)

@julio4 julio4 force-pushed the feat/iter_adapter_zip branch from 700421d to 2c7f719 Compare January 15, 2025 09:13
Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewed 1 of 1 files at r6, all commit messages.
Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)


a discussion (no related file):

Previously, orizi wrote…

@TomerStarkware for 2nd eye

ping.


corelib/src/iter/adapters/zip_adapter.cairo line 15 at r1 (raw file):

Previously, julio4 (Julio) wrote…

There's still some issue with default impl so I prefer not blocking thing and I'll add in a separate PR

Default impl should fully work now.

Copy link
Contributor Author

@julio4 julio4 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So should I add zip as a function then? I reverted it

Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)

Copy link
Contributor Author

@julio4 julio4 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)


corelib/src/iter/adapters/zip_adapter.cairo line 15 at r1 (raw file):

Previously, orizi wrote…

Default impl should fully work now.

Done.

@julio4 julio4 force-pushed the feat/iter_adapter_zip branch from 2c7f719 to 3e881f4 Compare January 15, 2025 12:01
Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewed 4 of 4 files at r7, all commit messages.
Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)

Copy link
Collaborator

@TomerStarkware TomerStarkware left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

:lgtm:

Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So should I add zip as a function then? I reverted it

lets hold on it for now.

Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)

Copy link
Collaborator

@orizi orizi left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewed 2 of 2 files at r8, all commit messages.
Reviewable status: all files reviewed, 1 unresolved discussion (waiting on @enitrat)

@orizi orizi enabled auto-merge January 15, 2025 16:13
@julio4 julio4 requested a review from enitrat January 17, 2025 15:58
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.

5 participants