nix-bindings 0.2347.3

Rust binding for Nix, the build tool
Documentation
use std::ptr::NonNull;

use crate::{Error, Result, Value, ValueType, sys};

impl Value<'_> {
  /// Check if this value is a list.
  ///
  /// # Example
  ///
  /// ```rust,no_run
  /// use std::sync::Arc;
  ///
  /// use nix_bindings::{Context, EvalStateBuilder, Store};
  /// fn main() -> Result<(), Box<dyn std::error::Error>> {
  ///   let ctx = Arc::new(Context::new()?);
  ///   let store = Arc::new(Store::open(&ctx, None)?);
  ///   let state = EvalStateBuilder::new(&store)?.build()?;
  ///   let list = state.eval_from_string("[1 2 3]", "<eval>")?;
  ///   assert!(list.is_list());
  ///   Ok(())
  /// }
  /// ```
  #[must_use]
  pub fn is_list(&self) -> bool {
    self.value_type() == ValueType::List
  }

  /// Get the length of this list.
  ///
  /// # Errors
  ///
  /// Returns an error if this value is not a list.
  ///
  /// # Example
  ///
  /// ```no_run
  /// # use std::sync::Arc;
  /// # use nix_bindings::{Context, EvalStateBuilder, Store};
  /// # fn main() -> Result<(), Box<dyn std::error::Error>> {
  /// # let ctx = Arc::new(Context::new()?);
  /// # let store = Arc::new(Store::open(&ctx, None)?);
  /// # let state = EvalStateBuilder::new(&store)?.build()?;
  /// let list = state.eval_from_string("[1 2 3]", "<eval>")?;
  /// assert_eq!(list.list_len()?, 3);
  /// # Ok(())
  /// # }
  /// ```
  pub fn list_len(&self) -> Result<usize> {
    if !self.is_list() {
      return Err(Error::InvalidType {
        expected: "list",
        actual:   self.value_type().to_string(),
      });
    }

    // SAFETY: context and value are valid, type is checked.
    // nix_get_list_size returns the length as c_uint with no error code.
    let len = unsafe {
      sys::nix_get_list_size(self.state.context.as_ptr(), self.inner.as_ptr())
    };

    Ok(len as usize)
  }

  /// Get an element from this list by index.
  ///
  /// # Arguments
  ///
  /// * `idx` - The index of the element to retrieve (0-based)
  ///
  /// # Errors
  ///
  /// Returns an error if this value is not a list or the index is out of
  /// bounds.
  ///
  /// # Example
  ///
  /// ```no_run
  /// # use std::sync::Arc;
  /// # use nix_bindings::{Context, EvalStateBuilder, Store};
  /// # fn main() -> Result<(), Box<dyn std::error::Error>> {
  /// # let ctx = Arc::new(Context::new()?);
  /// # let store = Arc::new(Store::open(&ctx, None)?);
  /// # let state = EvalStateBuilder::new(&store)?.build()?;
  /// let list = state.eval_from_string("[1 2 3]", "<eval>")?;
  /// let first = list.list_get(0)?;
  /// assert_eq!(first.as_int()?, 1);
  /// # Ok(())
  /// # }
  /// ```
  pub fn list_get(&self, idx: usize) -> Result<Value<'_>> {
    if !self.is_list() {
      return Err(Error::InvalidType {
        expected: "list",
        actual:   self.value_type().to_string(),
      });
    }

    let len = self.list_len()?;
    if idx >= len {
      return Err(Error::IndexOutOfBounds {
        index:  idx,
        length: len,
      });
    }

    // SAFETY: context, value, and state are valid; index is bounds-checked.
    // nix_get_list_byidx returns a GC-owned pointer (refcount incremented for
    // us). Value's Drop calls nix_value_decref to release our reference.
    let elem_ptr = unsafe {
      sys::nix_get_list_byidx(
        self.state.context.as_ptr(),
        self.inner.as_ptr(),
        self.state.as_ptr(),
        idx as std::os::raw::c_uint,
      )
    };

    let inner = NonNull::new(elem_ptr).ok_or(Error::NullPointer)?;
    Ok(Value {
      inner,
      state: self.state,
    })
  }

  /// Create an iterator over the elements of this list.
  ///
  /// # Errors
  ///
  /// Returns an error if this value is not a list.
  pub fn list_iter(&self) -> Result<ListIterator<'_>> {
    if !self.is_list() {
      return Err(Error::InvalidType {
        expected: "list",
        actual:   self.value_type().to_string(),
      });
    }

    let len = self.list_len()?;
    Ok(ListIterator {
      value:  self,
      index:  0,
      length: len,
    })
  }
}

/// Iterator over elements in a Nix list.
///
/// This struct is created by [`Value::list_iter`] and is used to iterate
/// over the elements of a Nix list.
#[derive(Debug)]
pub struct ListIterator<'a> {
  value:  &'a Value<'a>,
  index:  usize,
  length: usize,
}

impl<'a> Iterator for ListIterator<'a> {
  type Item = Result<Value<'a>>;

  fn next(&mut self) -> Option<Self::Item> {
    if self.index >= self.length {
      return None;
    }

    let idx = self.index;
    self.index += 1;

    // Bypass list_get's redundant type+length check; we already know the
    // value is a list and idx < length.
    // SAFETY: context, value, and state are valid; idx is bounds-checked.
    let elem_ptr = unsafe {
      sys::nix_get_list_byidx(
        self.value.state.context.as_ptr(),
        self.value.inner.as_ptr(),
        self.value.state.as_ptr(),
        idx as std::os::raw::c_uint,
      )
    };

    match NonNull::new(elem_ptr) {
      Some(inner) => {
        Some(Ok(Value {
          inner,
          state: self.value.state,
        }))
      },
      None => Some(Err(Error::NullPointer)),
    }
  }

  fn size_hint(&self) -> (usize, Option<usize>) {
    let remaining = self.length - self.index;
    (remaining, Some(remaining))
  }
}

impl ExactSizeIterator for ListIterator<'_> {}

#[cfg(test)]
mod tests {
  use std::sync::Arc;

  use serial_test::serial;

  use super::*;
  use crate::{Context, EvalStateBuilder, Store};

  #[test]
  #[serial]
  fn test_is_list() {
    let ctx = Arc::new(Context::new().expect("Failed to create context"));
    let store =
      Arc::new(Store::open(&ctx, None).expect("Failed to open store"));
    let state = EvalStateBuilder::new(&store)
      .expect("Failed to create builder")
      .build()
      .expect("Failed to build state");

    let list = state
      .eval_from_string("[1 2 3]", "<eval>")
      .expect("Failed to evaluate list");
    assert!(list.is_list());

    let int = state
      .eval_from_string("1", "<eval>")
      .expect("Failed to evaluate int");
    assert!(!int.is_list());
  }

  #[test]
  #[serial]
  fn test_list_len() {
    let ctx = Arc::new(Context::new().expect("Failed to create context"));
    let store =
      Arc::new(Store::open(&ctx, None).expect("Failed to open store"));
    let state = EvalStateBuilder::new(&store)
      .expect("Failed to create builder")
      .build()
      .expect("Failed to build state");

    let empty = state
      .eval_from_string("[]", "<eval>")
      .expect("Failed to evaluate empty list");
    assert_eq!(empty.list_len().expect("Failed to get list length"), 0);

    let list = state
      .eval_from_string("[1 2 3]", "<eval>")
      .expect("Failed to evaluate list");
    assert_eq!(list.list_len().expect("Failed to get list length"), 3);
  }

  #[test]
  #[serial]
  fn test_list_get() {
    let ctx = Arc::new(Context::new().expect("Failed to create context"));
    let store =
      Arc::new(Store::open(&ctx, None).expect("Failed to open store"));
    let state = EvalStateBuilder::new(&store)
      .expect("Failed to create builder")
      .build()
      .expect("Failed to build state");

    let list = state
      .eval_from_string("[10 20 30]", "<eval>")
      .expect("Failed to evaluate list");

    let first = list.list_get(0).expect("Failed to get first element");
    assert_eq!(first.as_int().expect("Failed to get int"), 10);

    let second = list.list_get(1).expect("Failed to get second element");
    assert_eq!(second.as_int().expect("Failed to get int"), 20);

    let third = list.list_get(2).expect("Failed to get third element");
    assert_eq!(third.as_int().expect("Failed to get int"), 30);

    // Out of bounds should return IndexOutOfBounds error
    let result = list.list_get(5);
    assert!(matches!(
      result,
      Err(Error::IndexOutOfBounds {
        index:  5,
        length: 3,
      })
    ));
  }

  #[test]
  #[serial]
  fn test_list_iter() {
    let ctx = Arc::new(Context::new().expect("Failed to create context"));
    let store =
      Arc::new(Store::open(&ctx, None).expect("Failed to open store"));
    let state = EvalStateBuilder::new(&store)
      .expect("Failed to create builder")
      .build()
      .expect("Failed to build state");

    let list = state
      .eval_from_string("[1 2 3]", "<eval>")
      .expect("Failed to evaluate list");

    let mut iter = list.list_iter().expect("Failed to create iterator");
    assert_eq!(iter.len(), 3);

    let first = iter
      .next()
      .expect("Failed to get first")
      .expect("Failed to get first value");
    assert_eq!(first.as_int().expect("Failed to get int"), 1);

    let second = iter
      .next()
      .expect("Failed to get second")
      .expect("Failed to get second value");
    assert_eq!(second.as_int().expect("Failed to get int"), 2);

    let third = iter
      .next()
      .expect("Failed to get third")
      .expect("Failed to get third value");
    assert_eq!(third.as_int().expect("Failed to get int"), 3);

    assert!(iter.next().is_none());
  }
}