scolapasta_aref/
lib.rs

1#![warn(clippy::all, clippy::pedantic, clippy::undocumented_unsafe_blocks)]
2#![allow(
3    clippy::let_underscore_untyped,
4    reason = "https://github.com/rust-lang/rust-clippy/pull/10442#issuecomment-1516570154"
5)]
6#![allow(
7    clippy::question_mark,
8    reason = "https://github.com/rust-lang/rust-clippy/issues/8281"
9)]
10#![allow(clippy::manual_let_else, reason = "manual_let_else was very buggy on release")]
11#![allow(clippy::missing_errors_doc, reason = "A lot of existing code fails this lint")]
12#![allow(
13    clippy::unnecessary_lazy_evaluations,
14    reason = "https://github.com/rust-lang/rust-clippy/issues/8109"
15)]
16#![cfg_attr(
17    test,
18    allow(clippy::non_ascii_literal, reason = "tests sometimes require UTF-8 string content")
19)]
20#![allow(unknown_lints)]
21#![warn(
22    missing_copy_implementations,
23    missing_debug_implementations,
24    missing_docs,
25    rust_2024_compatibility,
26    trivial_casts,
27    trivial_numeric_casts,
28    unused_qualifications,
29    variant_size_differences
30)]
31#![forbid(unsafe_code)]
32// Enable feature callouts in generated documentation:
33// https://doc.rust-lang.org/beta/unstable-book/language-features/doc-cfg.html
34//
35// This approach is borrowed from tokio.
36#![cfg_attr(docsrs, feature(doc_cfg))]
37#![cfg_attr(docsrs, feature(doc_alias))]
38
39//! Functions for working with Ruby containers that respond to `#[]` or "aref".
40//!
41//! # Examples
42//!
43//! Index into arrays:
44//!
45//! ```
46//! # fn example() -> Option<()> {
47//! let data = [1, 2, 3, 4, 5];
48//!
49//! // Positive offset
50//! let offset = 2;
51//! let index = scolapasta_aref::offset_to_index(offset, data.len())?;
52//! assert_eq!(index, 2);
53//! assert_eq!(data[index], 3);
54//!
55//! // Negative offset
56//! let offset = -3;
57//! let index = scolapasta_aref::offset_to_index(offset, data.len())?;
58//! assert_eq!(index, 2);
59//! assert_eq!(data[index], 3);
60//!
61//! // Out-of-bounds offset
62//! let offset = 10;
63//! let index = scolapasta_aref::offset_to_index(offset, data.len())?;
64//! assert_eq!(index, 10);
65//!
66//! // Out-of-bounds negative offset
67//! let offset = -10;
68//! let index = scolapasta_aref::offset_to_index(offset, data.len());
69//! assert_eq!(index, None);
70//! # Some(())
71//! # }
72//! # example().unwrap()
73//! ```
74//!
75//! Index into strings:
76//!
77//! ```
78//! # fn example() -> Option<()> {
79//! let data = "Hello, World!";
80//!
81//! // Positive offset
82//! let offset = 7;
83//! let index = scolapasta_aref::offset_to_index(offset, data.len())?;
84//! assert_eq!(index, 7);
85//! assert_eq!(&data[index..], "World!");
86//!
87//! // Negative offset
88//! let offset = -6;
89//! let index = scolapasta_aref::offset_to_index(offset, data.len())?;
90//! assert_eq!(index, 7);
91//! assert_eq!(&data[index..], "World!");
92//!
93//! // Out-of-bounds offset
94//! let offset = 20;
95//! let index = scolapasta_aref::offset_to_index(offset, data.len())?;
96//! assert_eq!(index, 20);
97//!
98//! // Out-of-bounds negative offset
99//! let offset = -20;
100//! let index = scolapasta_aref::offset_to_index(offset, data.len());
101//! assert_eq!(index, None);
102//! # Some(())
103//! # }
104//! # example().unwrap()
105//! ```
106
107#![no_std]
108
109// Ensure code blocks in `README.md` compile
110#[cfg(doctest)]
111#[doc = include_str!("../README.md")]
112mod readme {}
113
114/// Convert a signed aref offset to a `usize` index into the underlying container.
115///
116/// Negative indexes are interpreted as indexing from the end of the container
117/// as long as their magnitude is less than the given length.
118///
119/// Callers must still check whether the returned index is in bounds for the
120/// container. The returned index may be out of range since this routine can be
121/// used to calculate indexes beyond the length of the container during
122/// assignment (for example, `Array#[]=` may perform length-extension upon an
123/// out-of-bounds index).
124///
125/// # Examples
126///
127/// ```
128/// # fn example() -> Option<()> {
129/// let data = "ABC, 123, XYZ";
130///
131/// let offset = 6;
132/// let index = scolapasta_aref::offset_to_index(offset, data.len())?;
133/// assert_eq!(index, 6);
134/// assert_eq!(&data[index..], "23, XYZ");
135///
136/// let offset = 55;
137/// let index = scolapasta_aref::offset_to_index(offset, data.len())?;
138/// assert_eq!(index, 55);
139///
140/// let offset = -5;
141/// let index = scolapasta_aref::offset_to_index(offset, data.len())?;
142/// assert_eq!(index, 8);
143/// assert_eq!(&data[index..], ", XYZ");
144///
145/// let offset = -44;
146/// let index = scolapasta_aref::offset_to_index(offset, data.len());
147/// assert_eq!(index, None);
148/// # Some(())
149/// # }
150/// # example().unwrap()
151/// ```
152#[must_use]
153pub fn offset_to_index(index: i64, len: usize) -> Option<usize> {
154    // Here's an example of this behavior from `String`. All containers that
155    // respond to `#[]` ("aref") behave similarly.
156    //
157    // ```
158    // [3.0.1] > s = "abc"
159    // => "abc"
160    //
161    // [3.0.1] > s[-2]
162    // => "b"
163    // [3.0.1] > s[-3]
164    // => "a"
165    // [3.0.1] > s[-4]
166    // => nil
167    //
168    // [3.0.1] > s[-2, 10]
169    // => "bc"
170    // [3.0.1] > s[-3, 10]
171    // => "abc"
172    // [3.0.1] > s[-4, 10]
173    //
174    // [3.0.2] > s.byteslice(-2, 10)
175    // => "bc"
176    // [3.0.2] > s.byteslice(-3, 10)
177    // => "abc"
178    // [3.0.2] > s.byteslice(-4, 10)
179    // => nil
180    // => nil
181    // ```
182    match usize::try_from(index) {
183        Ok(index) => Some(index),
184        Err(_) => index
185            .checked_neg()
186            .and_then(|index| usize::try_from(index).ok())
187            .and_then(|index| len.checked_sub(index)),
188    }
189}
190
191#[cfg(test)]
192mod tests {
193    use super::*;
194
195    #[test]
196    fn test_zero_index() {
197        // Test case: `index = 0, len = 0`
198        assert_eq!(offset_to_index(0_i64, 0_usize), Some(0_usize));
199
200        // Test case: `index = 0, len = 1`
201        assert_eq!(offset_to_index(0_i64, 1_usize), Some(0_usize));
202
203        // Test case: `index = 0, len = 10`
204        assert_eq!(offset_to_index(0, 10), Some(0_usize));
205
206        // Test case: `index = 0, len = usize::MAX`
207        assert_eq!(offset_to_index(0_i64, usize::MAX), Some(0_usize));
208    }
209
210    #[test]
211    fn test_positive_index() {
212        // Test case: `index = 1, len = 0`
213        assert_eq!(offset_to_index(1_i64, 0_usize), Some(1_usize));
214
215        // Test case: `index = 1, len = 1`
216        assert_eq!(offset_to_index(1_i64, 1_usize), Some(1_usize));
217
218        // Test case: `index = 1, len = 2`
219        assert_eq!(offset_to_index(1_i64, 2_usize), Some(1_usize));
220
221        // Test case: `index = 1, len = usize::MAX`
222        assert_eq!(offset_to_index(1_i64, usize::MAX), Some(1_usize));
223
224        // Test case: `index = 15, len = 10`
225        assert_eq!(offset_to_index(15, 10), Some(15_usize));
226
227        // Test case: `index = 123, len = 0`
228        assert_eq!(offset_to_index(123_i64, 0_usize), Some(123_usize));
229
230        // Test case: `index = 123, len = 1`
231        assert_eq!(offset_to_index(123_i64, 1_usize), Some(123_usize));
232
233        // Test case: `index = 123, len = 123`
234        assert_eq!(offset_to_index(123_i64, 123_usize), Some(123_usize));
235
236        // Test case: `index = 123, len = 123`
237        assert_eq!(offset_to_index(123_i64, 124_usize), Some(123_usize));
238
239        // Test case: `index = 123, len = 123`
240        assert_eq!(offset_to_index(123_i64, 500_usize), Some(123_usize));
241
242        // Test case: `index = 123, len = usize::MAX`
243        assert_eq!(offset_to_index(123_i64, usize::MAX), Some(123_usize));
244
245        // Test case: `index = i64::MAX, len = 5`
246        #[cfg(target_pointer_width = "64")]
247        assert_eq!(offset_to_index(i64::MAX, 5), Some(usize::try_from(i64::MAX).unwrap()));
248
249        // Test case: `index = i64::MAX, len = usize::MAX`
250        #[cfg(target_pointer_width = "64")]
251        assert_eq!(
252            offset_to_index(i64::MAX, usize::MAX),
253            Some(usize::try_from(i64::MAX).unwrap())
254        );
255
256        // Test case: `index = 100, len = 1000`
257        assert_eq!(offset_to_index(100_i64, 1000_usize), Some(100_usize));
258
259        // Test case: `index = 500, len = 500`
260        assert_eq!(offset_to_index(500_i64, 500_usize), Some(500_usize));
261
262        // Test case: `index = 999, len = 100`
263        assert_eq!(offset_to_index(999_i64, 100_usize), Some(999_usize));
264    }
265
266    #[test]
267    fn test_negative_index() {
268        // Test case: `index = -1, len = 0`
269        assert_eq!(offset_to_index(-1_i64, 0_usize), None);
270
271        // Test case: `index = -1, len = 1`
272        assert_eq!(offset_to_index(-1_i64, 1_usize), Some(0_usize));
273
274        // Test case: `index = -1, len = 2`
275        assert_eq!(offset_to_index(-1_i64, 2_usize), Some(1_usize));
276
277        // Test case: `index = -1, len = 10`
278        assert_eq!(offset_to_index(-1_i64, 10_usize), Some(9_usize));
279
280        // Test case: `index = -1, len = 245`
281        assert_eq!(offset_to_index(-1_i64, 245_usize), Some(244_usize));
282
283        // Test case: `index = -10, len = 0`
284        assert_eq!(offset_to_index(-10_i64, 0_usize), None);
285
286        // Test case: `index = -10, len = 1`
287        assert_eq!(offset_to_index(-10_i64, 1_usize), None);
288
289        // Test case: `index = -10, len = 2`
290        assert_eq!(offset_to_index(-10_i64, 2_usize), None);
291
292        // Test case: `index = -10, len = 10`
293        assert_eq!(offset_to_index(-10_i64, 10_usize), Some(0_usize));
294
295        // Test case: `index = -10, len = 245`
296        assert_eq!(offset_to_index(-10_i64, 245_usize), Some(235_usize));
297
298        // Test case: `index = -123, len = 0`
299        assert_eq!(offset_to_index(-123_i64, 0_usize), None);
300
301        // Test case: `index = -123, len = 1`
302        assert_eq!(offset_to_index(-123_i64, 1_usize), None);
303
304        // Test case: `index = -123, len = 2`
305        assert_eq!(offset_to_index(-123_i64, 2_usize), None);
306
307        // Test case: `index = -123, len = 10`
308        assert_eq!(offset_to_index(-123_i64, 10_usize), None);
309
310        // Test case: `index = -123, len = 245`
311        assert_eq!(offset_to_index(-123_i64, 245_usize), Some(122_usize));
312
313        // Test case: `index = i64::MIN, len = 0`
314        assert_eq!(offset_to_index(i64::MIN, 0_usize), None);
315
316        // Test case: `index = i64::MIN, len = 1`
317        assert_eq!(offset_to_index(i64::MIN, 1_usize), None);
318
319        // Test case: `index = i64::MIN, len = 2`
320        assert_eq!(offset_to_index(i64::MIN, 2_usize), None);
321
322        // Test case: `index = i64::MIN, len = 10`
323        assert_eq!(offset_to_index(i64::MIN, 10_usize), None);
324
325        // Test case: `index = i64::MIN, len = 245`
326        assert_eq!(offset_to_index(i64::MIN, 245_usize), None);
327    }
328
329    #[test]
330    fn test_out_of_bounds_positive_offset() {
331        // Test case: Offset greater than or equal to length
332        //
333        // ```
334        // [3.2.2] > a = [1,2,3,4,5]
335        // => [1, 2, 3, 4, 5]
336        // [3.2.2] > a[10]
337        // => nil
338        // [3.2.2] > a[10] = 'a'
339        // => "a"
340        // [3.2.2] > a
341        // => [1, 2, 3, 4, 5, nil, nil, nil, nil, nil, "a"]
342        // ```
343        assert_eq!(offset_to_index(10, 5), Some(10_usize));
344    }
345
346    #[test]
347    fn test_positive_offset_equal_to_length() {
348        // ```
349        // [3.2.2] > a = [1,2,3,4,5]
350        // => [1, 2, 3, 4, 5]
351        // [3.2.2] > a[5]
352        // => nil
353        // [3.2.2] > a[5, 0]
354        // => []
355        // [3.2.2] > a[5] = 'a'
356        // => "a"
357        // [3.2.2] > a
358        // => [1, 2, 3, 4, 5, "a"]
359        // ```
360        assert_eq!(offset_to_index(5, 5), Some(5_usize));
361    }
362
363    #[test]
364    fn test_negative_offset_of_magnitude_length() {
365        // Test case: Offset equal to negative length
366        //
367        // ```
368        // [3.2.2] > a = [1,2,3,4,5]
369        // => [1, 2, 3, 4, 5]
370        // [3.2.2] > a[-5]
371        // => 1
372        // [3.2.2] > a[-5] = 'a'
373        // => "a"
374        // [3.2.2] > a
375        // => ["a", 2, 3, 4, 5]
376        // ```
377        assert_eq!(offset_to_index(-5, 5), Some(0));
378
379        assert_eq!(offset_to_index(-10, 10), Some(0_usize));
380    }
381
382    #[test]
383    fn test_invalid_negative_offset() {
384        // Test case: Offset less than negative length
385        //
386        // ```
387        // [3.2.2] > a = [1,2,3,4,5]
388        // => [1, 2, 3, 4, 5]
389        // [3.2.2] > a[-10]
390        // => nil
391        // [3.2.2] > a[-10] = 'a'
392        // (irb):5:in `<main>': index -10 too small for array; minimum: -5 (IndexError)
393        // ```
394        assert_eq!(offset_to_index(-10, 5), None);
395    }
396
397    #[test]
398    fn test_edge_cases() {
399        // Test case: Length is zero
400        assert_eq!(offset_to_index(0, 0), Some(0_usize));
401
402        // Test case: Offset is the minimum `i64` value
403        assert_eq!(offset_to_index(i64::MIN, 10), None);
404
405        // Test case: Offset is the maximum `i64` value
406        #[cfg(target_pointer_width = "64")]
407        assert_eq!(offset_to_index(i64::MAX, 10), Some(usize::try_from(i64::MAX).unwrap()));
408
409        // Test case: `index = 0, len = usize::MAX`
410        assert_eq!(offset_to_index(0_i64, usize::MAX), Some(0_usize));
411
412        // Test case: `index = 1, len = usize::MAX`
413        assert_eq!(offset_to_index(1_i64, usize::MAX), Some(1_usize));
414
415        // Test case: `index = -1, len = usize::MAX`
416        assert_eq!(offset_to_index(-1_i64, usize::MAX), Some(usize::MAX - 1));
417
418        // Test case: `index = 10, len = usize::MAX`
419        assert_eq!(offset_to_index(10, usize::MAX), Some(10_usize));
420
421        // Test case: `index = i64::MAX, len = usize::MAX`
422        #[cfg(target_pointer_width = "64")]
423        assert_eq!(
424            offset_to_index(i64::MAX, usize::MAX),
425            Some(usize::try_from(i64::MAX).unwrap())
426        );
427    }
428}