winnow/stream/
locating.rs

1use crate::error::Needed;
2use crate::stream::AsBStr;
3use crate::stream::AsBytes;
4use crate::stream::Checkpoint;
5use crate::stream::Compare;
6use crate::stream::CompareResult;
7use crate::stream::FindSlice;
8use crate::stream::Location;
9use crate::stream::Offset;
10#[cfg(feature = "unstable-recover")]
11#[cfg(feature = "std")]
12use crate::stream::Recover;
13use crate::stream::SliceLen;
14use crate::stream::Stream;
15use crate::stream::StreamIsPartial;
16use crate::stream::UpdateSlice;
17
18/// Allow collecting the span of a parsed token within a slice
19///
20/// Converting byte offsets to line or column numbers is left up to the user, as computing column
21/// numbers requires domain knowledge (are columns byte-based, codepoint-based, or grapheme-based?)
22/// and O(n) iteration over the input to determine codepoint and line boundaries.
23///
24/// [The `line-span` crate](https://docs.rs/line-span/latest/line_span/) can help with converting
25/// byte offsets to line numbers.
26///
27/// See [`Parser::span`][crate::Parser::span] and [`Parser::with_span`][crate::Parser::with_span] for more details
28#[derive(Copy, Clone, Default, Debug, PartialEq, Eq, PartialOrd, Ord)]
29#[doc(alias = "LocatingSliceSpan")]
30#[doc(alias = "Located")]
31pub struct LocatingSlice<I> {
32    initial: I,
33    input: I,
34}
35
36impl<I> LocatingSlice<I>
37where
38    I: Clone + Offset,
39{
40    /// Wrap another Stream with span tracking
41    pub fn new(input: I) -> Self {
42        let initial = input.clone();
43        Self { initial, input }
44    }
45
46    #[inline]
47    fn previous_token_end(&self) -> usize {
48        // Assumptions:
49        // - Index offsets is sufficient
50        // - Tokens are continuous
51        self.input.offset_from(&self.initial)
52    }
53    #[inline]
54    fn current_token_start(&self) -> usize {
55        // Assumptions:
56        // - Index offsets is sufficient
57        self.input.offset_from(&self.initial)
58    }
59}
60
61impl<I> LocatingSlice<I>
62where
63    I: Clone + Stream + Offset,
64{
65    /// Reset the stream to the start
66    ///
67    /// This is useful for formats that encode a graph with addresses relative to the start of the
68    /// input.
69    #[doc(alias = "fseek")]
70    #[inline]
71    pub fn reset_to_start(&mut self) {
72        let start = self.initial.checkpoint();
73        self.input.reset(&start);
74    }
75}
76
77impl<I> AsRef<I> for LocatingSlice<I> {
78    #[inline(always)]
79    fn as_ref(&self) -> &I {
80        &self.input
81    }
82}
83
84impl<I> crate::lib::std::ops::Deref for LocatingSlice<I> {
85    type Target = I;
86
87    #[inline(always)]
88    fn deref(&self) -> &Self::Target {
89        &self.input
90    }
91}
92
93impl<I: crate::lib::std::fmt::Display> crate::lib::std::fmt::Display for LocatingSlice<I> {
94    fn fmt(&self, f: &mut crate::lib::std::fmt::Formatter<'_>) -> crate::lib::std::fmt::Result {
95        self.input.fmt(f)
96    }
97}
98
99impl<I> SliceLen for LocatingSlice<I>
100where
101    I: SliceLen,
102{
103    #[inline(always)]
104    fn slice_len(&self) -> usize {
105        self.input.slice_len()
106    }
107}
108
109impl<I: Stream> Stream for LocatingSlice<I> {
110    type Token = <I as Stream>::Token;
111    type Slice = <I as Stream>::Slice;
112
113    type IterOffsets = <I as Stream>::IterOffsets;
114
115    type Checkpoint = Checkpoint<I::Checkpoint, Self>;
116
117    #[inline(always)]
118    fn iter_offsets(&self) -> Self::IterOffsets {
119        self.input.iter_offsets()
120    }
121    #[inline(always)]
122    fn eof_offset(&self) -> usize {
123        self.input.eof_offset()
124    }
125
126    #[inline(always)]
127    fn next_token(&mut self) -> Option<Self::Token> {
128        self.input.next_token()
129    }
130
131    #[inline(always)]
132    fn peek_token(&self) -> Option<Self::Token> {
133        self.input.peek_token()
134    }
135
136    #[inline(always)]
137    fn offset_for<P>(&self, predicate: P) -> Option<usize>
138    where
139        P: Fn(Self::Token) -> bool,
140    {
141        self.input.offset_for(predicate)
142    }
143    #[inline(always)]
144    fn offset_at(&self, tokens: usize) -> Result<usize, Needed> {
145        self.input.offset_at(tokens)
146    }
147    #[inline(always)]
148    fn next_slice(&mut self, offset: usize) -> Self::Slice {
149        self.input.next_slice(offset)
150    }
151    #[inline(always)]
152    fn peek_slice(&self, offset: usize) -> Self::Slice {
153        self.input.peek_slice(offset)
154    }
155
156    #[inline(always)]
157    fn checkpoint(&self) -> Self::Checkpoint {
158        Checkpoint::<_, Self>::new(self.input.checkpoint())
159    }
160    #[inline(always)]
161    fn reset(&mut self, checkpoint: &Self::Checkpoint) {
162        self.input.reset(&checkpoint.inner);
163    }
164
165    #[inline(always)]
166    fn raw(&self) -> &dyn crate::lib::std::fmt::Debug {
167        &self.input
168    }
169}
170
171impl<I> Location for LocatingSlice<I>
172where
173    I: Clone + Offset,
174{
175    #[inline(always)]
176    fn previous_token_end(&self) -> usize {
177        self.previous_token_end()
178    }
179    #[inline(always)]
180    fn current_token_start(&self) -> usize {
181        self.current_token_start()
182    }
183}
184
185#[cfg(feature = "unstable-recover")]
186#[cfg(feature = "std")]
187impl<I, E> Recover<E> for LocatingSlice<I>
188where
189    I: Recover<E>,
190    I: Stream,
191{
192    #[inline(always)]
193    fn record_err(
194        &mut self,
195        _token_start: &Self::Checkpoint,
196        _err_start: &Self::Checkpoint,
197        err: E,
198    ) -> Result<(), E> {
199        Err(err)
200    }
201
202    /// Report whether the [`Stream`] can save off errors for recovery
203    #[inline(always)]
204    fn is_recovery_supported() -> bool {
205        false
206    }
207}
208
209impl<I> StreamIsPartial for LocatingSlice<I>
210where
211    I: StreamIsPartial,
212{
213    type PartialState = I::PartialState;
214
215    #[inline]
216    fn complete(&mut self) -> Self::PartialState {
217        self.input.complete()
218    }
219
220    #[inline]
221    fn restore_partial(&mut self, state: Self::PartialState) {
222        self.input.restore_partial(state);
223    }
224
225    #[inline(always)]
226    fn is_partial_supported() -> bool {
227        I::is_partial_supported()
228    }
229
230    #[inline(always)]
231    fn is_partial(&self) -> bool {
232        self.input.is_partial()
233    }
234}
235
236impl<I> Offset for LocatingSlice<I>
237where
238    I: Stream,
239{
240    #[inline(always)]
241    fn offset_from(&self, other: &Self) -> usize {
242        self.offset_from(&other.checkpoint())
243    }
244}
245
246impl<I> Offset<<LocatingSlice<I> as Stream>::Checkpoint> for LocatingSlice<I>
247where
248    I: Stream,
249{
250    #[inline(always)]
251    fn offset_from(&self, other: &<LocatingSlice<I> as Stream>::Checkpoint) -> usize {
252        self.checkpoint().offset_from(other)
253    }
254}
255
256impl<I> AsBytes for LocatingSlice<I>
257where
258    I: AsBytes,
259{
260    #[inline(always)]
261    fn as_bytes(&self) -> &[u8] {
262        self.input.as_bytes()
263    }
264}
265
266impl<I> AsBStr for LocatingSlice<I>
267where
268    I: AsBStr,
269{
270    #[inline(always)]
271    fn as_bstr(&self) -> &[u8] {
272        self.input.as_bstr()
273    }
274}
275
276impl<I, U> Compare<U> for LocatingSlice<I>
277where
278    I: Compare<U>,
279{
280    #[inline(always)]
281    fn compare(&self, other: U) -> CompareResult {
282        self.input.compare(other)
283    }
284}
285
286impl<I, T> FindSlice<T> for LocatingSlice<I>
287where
288    I: FindSlice<T>,
289{
290    #[inline(always)]
291    fn find_slice(&self, substr: T) -> Option<crate::lib::std::ops::Range<usize>> {
292        self.input.find_slice(substr)
293    }
294}
295
296impl<I> UpdateSlice for LocatingSlice<I>
297where
298    I: UpdateSlice,
299{
300    #[inline(always)]
301    fn update_slice(mut self, inner: Self::Slice) -> Self {
302        self.input = I::update_slice(self.input, inner);
303        self
304    }
305}