Closed joshka closed 1 month ago
This is a duplicate of https://github.com/rust-itertools/itertools/issues/942, which is tracked by strum in https://github.com/Peternator7/strum/issues/358, and will be fixed by https://github.com/Peternator7/strum/pull/357.
Ah thanks - apologies for not searching.
In https://github.com/ratatui-org/ratatui/pull/1120 dependabot brought in itertools 0.13. We have a dependency on strum and use the EnumIter derive macro. This fails to compile with itertools due to adding the
.get
method on the iterator (which clashes with an existing method on the iterator created by the derive macro.)E.g.:
Errors
``` error[E0277]: the trait bound `usize: IteratorIndex<&mut ConstraintNameIter>` is not satisfied --> examples/constraint-explorer.rs:53:54 | 53 | #[derive(Debug, Default, Clone, Copy, PartialEq, Eq, EnumIter, FromRepr, Display)] | ^^^^^^^^ the trait `IteratorIndex<&mut ConstraintNameIter>` is not implemented for `usize` | = help: the following other types implement trait `IteratorIndex`: RangeFull std::ops::Range
RangeFrom
RangeTo
RangeInclusive
RangeToInclusive
note: required by a bound in `itertools::Itertools::get`
--> /Users/joshka/.cargo/registry/src/index.crates.io-6f17d22bba15001f/itertools-0.13.0/src/lib.rs:554:12
|
551 | fn get(self, index: R) -> R::Output
| --- required by a bound in this associated function
...
554 | R: traits::IteratorIndex,
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^ required by this bound in `Itertools::get`
= note: this error originates in the derive macro `EnumIter` (in Nightly builds, run with -Z macro-backtrace for more info)
error[E0277]: the trait bound `usize: IteratorIndex<&mut TabIter>` is not satisfied
--> examples/demo2/app.rs:30:48
|
30 | #[derive(Debug, Clone, Copy, Default, Display, EnumIter, FromRepr, PartialEq, Eq)]
| ^^^^^^^^ the trait `IteratorIndex<&mut TabIter>` is not implemented for `usize`
|
= help: the following other types implement trait `IteratorIndex`:
RangeFull
std::ops::Range
RangeFrom
RangeTo
RangeInclusive
RangeToInclusive
note: required by a bound in `itertools::Itertools::get`
--> /Users/joshka/.cargo/registry/src/index.crates.io-6f17d22bba15001f/itertools-0.13.0/src/lib.rs:554:12
|
551 | fn get(self, index: R) -> R::Output
| --- required by a bound in this associated function
...
554 | R: traits::IteratorIndex,
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^ required by this bound in `Itertools::get`
= note: this error originates in the derive macro `EnumIter` (in Nightly builds, run with -Z macro-backtrace for more info)
For more information about this error, try `rustc --explain E0277`.
error: could not compile `ratatui` (example "constraint-explorer") due to 2 previous errors
warning: build failed, waiting for other jobs to finish...
error: could not compile `ratatui` (example "demo2") due to 2 previous errors
```
EnumIter macro expansion
```rust // Recursive expansion of EnumIter macro // ====================================== #[doc = "An iterator over the variants of [Tab]"] #[allow(missing_copy_implementations)] struct TabIter { idx: usize, back_idx: usize, marker: ::core::marker::PhantomData<()>, } impl ::core::fmt::Debug for TabIter { fn fmt(&self, f: &mut ::core::fmt::Formatter<'_>) -> ::core::fmt::Result { f.debug_struct("TabIter").field("len", &self.len()).finish() } } impl TabIter { fn get(&self, idx: usize) -> ::core::option::Option {
match idx {
0usize => ::core::option::Option::Some(Tab::About),
1usize => ::core::option::Option::Some(Tab::Recipe),
2usize => ::core::option::Option::Some(Tab::Email),
3usize => ::core::option::Option::Some(Tab::Traceroute),
4usize => ::core::option::Option::Some(Tab::Weather),
_ => ::core::option::Option::None,
}
}
}
impl ::strum::IntoEnumIterator for Tab {
type Iterator = TabIter;
fn iter() -> TabIter {
TabIter {
idx: 0,
back_idx: 0,
marker: ::core::marker::PhantomData,
}
}
}
impl Iterator for TabIter {
type Item = Tab;
fn next(&mut self) -> ::core::option::Option<::Item> {
self.nth(0)
}
fn size_hint(&self) -> (usize, ::core::option::Option) {
let t = if self.idx + self.back_idx >= 5usize {
0
} else {
5usize - self.idx - self.back_idx
};
(t, Some(t))
}
fn nth(&mut self, n: usize) -> ::core::option::Option<::Item> {
let idx = self.idx + n + 1;
if idx + self.back_idx > 5usize {
self.idx = 5usize;
::core::option::Option::None
} else {
self.idx = idx;
self.get(idx - 1)
}
}
}
impl ExactSizeIterator for TabIter {
fn len(&self) -> usize {
self.size_hint().0
}
}
impl DoubleEndedIterator for TabIter {
fn next_back(&mut self) -> ::core::option::Option<::Item> {
let back_idx = self.back_idx + 1;
if self.idx + back_idx > 5usize {
self.back_idx = 5usize;
::core::option::Option::None
} else {
self.back_idx = back_idx;
self.get(5usize - self.back_idx)
}
}
}
impl ::core::iter::FusedIterator for TabIter {}
impl Clone for TabIter {
fn clone(&self) -> TabIter {
TabIter {
idx: self.idx,
back_idx: self.back_idx,
marker: self.marker.clone(),
}
}
}
```
I can work around this by just manually implementing the EnumIter trait from strum (it's an small shortcut), but I'd guess that this was unintentional. Naming a method
get
on a widely used extension trait intuitively seems like it might cause similar issues in many places. I'm not certain of this though, and it's possible I'm over-generalizing the failure mode from this one instance.Caused by https://github.com/rust-itertools/itertools/pull/891 and https://github.com/rust-itertools/itertools/pull/447