tskit 0.15.0

rust interface to tskit
Documentation
#[test]
fn simple_table_collection_creation_with_newtype() {
    // ANCHOR: create_table_collection_with_newtype
    let sequence_length = tskit::Position::from(100.0);
    if let Ok(tables) = tskit::TableCollection::new(sequence_length) {
        assert_eq!(tables.sequence_length(), sequence_length);
        // In tskit, the various newtypes can be compared to
        // the low-level types they wrap.
        assert_eq!(tables.sequence_length(), 100.0);
    } else {
        panic!(
            "TableCollection creation sequence length = {} failed",
            sequence_length
        );
    }
    // ANCHOR_END: create_table_collection_with_newtype
}

#[test]
fn simple_table_collection_creation() {
    // ANCHOR: create_table_collection
    let tables = tskit::TableCollection::new(100.0).unwrap();
    // ANCHOR_END: create_table_collection
    assert_eq!(tables.sequence_length(), 100.0);
}

#[test]
fn add_node_without_metadata() {
    {
        // ANCHOR: add_node_without_metadata
        let mut tables = tskit::TableCollection::new(100.0).unwrap();
        if let Ok(node_id) = tables.add_node(
            0,                         // Node flags
            tskit::Time::from(0.0),    // Birth time
            tskit::PopulationId::NULL, // Population id
            tskit::IndividualId::NULL, // Individual id
        ) {
            assert_eq!(node_id, 0);
        }
        // ANCHOR_END: add_node_without_metadata
    }
    {
        let mut tables = tskit::TableCollection::new(100.0).unwrap();
        // ANCHOR: add_node_without_metadata_using_into
        let node_id = tables.add_node(0, 0.0, -1, -1).unwrap();
        // ANCHOR_END: add_node_without_metadata_using_into
        assert_eq!(node_id, 0);
    }
}

#[test]
fn add_node_handle_error() {
    // ANCHOR: integrity_check
    let mut tables = tskit::TableCollection::new(100.0).unwrap();
    // Everything about this edge is wrong...
    tables.add_edge(-1.0, 110.0, 0, 1).unwrap();
    // ...and we can catch that here
    match tables.check_integrity(tskit::TableIntegrityCheckFlags::default()) {
        Ok(code) => panic!("expected Err(e) but got code: {}", code),
        // tskit::TskitError can be formatted into the same
        // error messages that tskit-c/tskit-python give.
        Err(e) => println!("{}", e),
    }
    // ANCHOR_END: integrity_check
    assert!(tables
        .check_integrity(tskit::TableIntegrityCheckFlags::default())
        .is_err());
}

#[test]
fn get_data_from_edge_table() {
    use rand::distributions::Distribution;
    use tskit::prelude::*;
    let sequence_length = tskit::Position::from(100.0);
    let mut rng = rand::thread_rng();
    let random_pos = rand::distributions::Uniform::new::<f64, f64>(0., sequence_length.into());
    let mut tables = tskit::TableCollection::new(sequence_length).unwrap();
    let child = tables.add_node(0, 0.0, -1, -1).unwrap();
    let parent = tables.add_node(0, 1.0, -1, -1).unwrap();
    let mut left = tskit::Position::from(random_pos.sample(&mut rng));
    let mut right = tskit::Position::from(random_pos.sample(&mut rng));
    if left > right {
        std::mem::swap(&mut left, &mut right);
    }

    // ANCHOR: get_edge_table_columns
    if let Ok(edge_id) = tables.add_edge(left, right, parent, child) {
        // Take a reference to an edge table (& tskit::EdgeTable)
        let edges = tables.edges();
        if let Some(p) = edges.parent(edge_id) {
            assert_eq!(p, parent);
        }
        if let Some(c) = edges.child(edge_id) {
            assert_eq!(c, child);
        }
        if let Some(l) = edges.left(edge_id) {
            assert_eq!(l, left);
        }
        if let Some(r) = edges.right(edge_id) {
            assert_eq!(r, right);
        }
    } else {
        panic!("that should have worked...");
    }
    // ANCHOR_END: get_edge_table_columns

    // ANCHOR: get_edge_table_columns_out_of_range
    assert!(tables.edges().parent(tskit::EdgeId::NULL).is_none());
    // ANCHOR_END: get_edge_table_columns_out_of_range

    let edge_id = tskit::EdgeId::from(0);
    // ANCHOR: get_edge_table_row_by_id
    if let Some(row) = tables.edges().row(edge_id) {
        assert_eq!(row.id, 0);
        assert_eq!(row.left, left);
        assert_eq!(row.right, right);
        assert_eq!(row.parent, parent);
        assert_eq!(row.child, child);
    } else {
        panic!("that should have worked...");
    }
    // ANCHOR_END: get_edge_table_row_by_id

    // ANCHOR: get_edge_table_row_view_by_id
    if let Some(row_view) = tables.edges().row_view(edge_id) {
        assert_eq!(row_view.id, 0);
        assert_eq!(row_view.left, left);
        assert_eq!(row_view.right, right);
        assert_eq!(row_view.parent, parent);
        assert_eq!(row_view.child, child);
    } else {
        panic!("that should have worked...");
    }
    // ANCHOR_END: get_edge_table_row_view_by_id

    // ANCHOR: get_edge_table_rows_by_lending_iterator
    let mut edge_table_lending_iter = tables.edges().lending_iter();
    while let Some(row_view) = edge_table_lending_iter.next() {
        // there is only one row!
        assert_eq!(row_view.id, 0);
        assert_eq!(row_view.left, left);
        assert_eq!(row_view.right, right);
        assert_eq!(row_view.parent, parent);
        assert_eq!(row_view.child, child);
        assert!(row_view.metadata.is_none()); // no metadata in our table
    }
    // ANCHOR_END: get_edge_table_rows_by_lending_iterator

    assert!(tables
        .check_integrity(tskit::TableIntegrityCheckFlags::default())
        .is_ok());

    // ANCHOR: get_edge_table_rows_by_iterator
    for row in tables.edges_iter() {
        // there is only one row!
        assert_eq!(row.id, 0);
        assert_eq!(row.left, left);
        assert_eq!(row.right, right);
        assert_eq!(row.parent, parent);
        assert_eq!(row.child, child);
    }
    // ANCHOR_END: get_edge_table_rows_by_iterator

    assert!(tables
        .check_integrity(tskit::TableIntegrityCheckFlags::default())
        .is_ok());
}

#[test]
fn test_adding_node_table_row_with_defaults() {
    let mut tables = tskit::TableCollection::new(10.).unwrap();
    // ANCHOR: node_defaults
    let defaults = tskit::NodeDefaults::default();
    // ANCHOR_END: node_defaults
    // ANCHOR: add_node_defaults
    let node = tables.add_node_with_defaults(0.0, &defaults).unwrap();
    // ANCHOR_END: add_node_defaults
    assert_eq!(node, 0);

    // ANCHOR: add_node_defaults_sample
    let node = tables
        .add_node_with_defaults(
            0.0,
            // Create a new, temporary defaults instance
            &tskit::NodeDefaults {
                // Mark the new node as a sample
                flags: tskit::NodeFlags::new_sample(),
                // Use remaining values from our current defaults
                ..defaults
            },
        )
        .unwrap();
    // ANCHOR_END: add_node_defaults_sample
    assert!(tables.nodes().flags(node).unwrap().is_sample());
}

macro_rules! impl_node_metadata_traits {
    () => {
        impl tskit::metadata::MetadataRoundtrip for NodeMetadata {
            fn encode(&self) -> Result<Vec<u8>, tskit::metadata::MetadataError> {
                match serde_json::to_string(self) {
                    Ok(x) => Ok(x.as_bytes().to_vec()),
                    Err(e) => {
                        Err(::tskit::metadata::MetadataError::RoundtripError { value: Box::new(e) })
                    }
                }
            }
            fn decode(md: &[u8]) -> Result<Self, tskit::metadata::MetadataError>
            where
                Self: Sized,
            {
                match serde_json::from_slice(md) {
                    Ok(v) => Ok(v),
                    Err(e) => {
                        Err(::tskit::metadata::MetadataError::RoundtripError { value: Box::new(e) })
                    }
                }
            }
        }
        impl tskit::metadata::NodeMetadata for NodeMetadata {}
    };
}

mod node_metadata {
    #[derive(Debug, serde::Serialize, serde::Deserialize, Eq, PartialEq)]
    // ANCHOR: node_metadata
    pub struct NodeMetadata {
        pub value: i32,
    }
    // ANCHOR_END: node_metadata
    impl_node_metadata_traits!();
}

mod node_metadata_clone {
    #[derive(Debug, Eq, PartialEq, Clone, serde::Serialize, serde::Deserialize)]
    pub struct NodeMetadata {
        pub value: i32,
    }
    impl_node_metadata_traits!();
}

#[test]
fn test_adding_node_table_row_with_defaults_and_metadata() {
    use node_metadata::NodeMetadata;
    let mut tables = tskit::TableCollection::new(10.0).unwrap();
    // ANCHOR: node_defaults_with_metadata

    // Create a type alias for brevity
    type DefaultsWithMetadata = tskit::NodeDefaultsWithMetadata<NodeMetadata>;
    // Default metadata is None
    let defaults = DefaultsWithMetadata::default();

    // A row with no metadata
    let n0 = tables.add_node_with_defaults(0.0, &defaults).unwrap();

    // A row with metadata
    let n1 = tables
        .add_node_with_defaults(
            0.0,
            &DefaultsWithMetadata {
                population: 3.into(),
                metadata: Some(NodeMetadata { value: 42 }),
                ..defaults
            },
        )
        .unwrap();

    // Another row with metadata, different from the last.
    let n2 = tables
        .add_node_with_defaults(
            0.0,
            &DefaultsWithMetadata {
                population: 1.into(),
                metadata: Some(NodeMetadata { value: 1234 }),
                ..defaults
            },
        )
        .unwrap();
    // ANCHOR_END: node_defaults_with_metadata
    assert!(tables.nodes().metadata::<NodeMetadata>(n0).is_none());
    assert_eq!(
        tables
            .nodes()
            .metadata::<NodeMetadata>(n1)
            .unwrap()
            .unwrap(),
        NodeMetadata { value: 42 }
    );
    assert_eq!(
        tables
            .nodes()
            .metadata::<NodeMetadata>(n2)
            .unwrap()
            .unwrap(),
        NodeMetadata { value: 1234 }
    );
}

#[test]
fn test_adding_node_table_row_with_defaults_and_metadata_requiring_clone() {
    use node_metadata_clone::NodeMetadata;
    let mut tables = tskit::TableCollection::new(10.0).unwrap();
    type DefaultsWithMetadata = tskit::NodeDefaultsWithMetadata<NodeMetadata>;

    // ANCHOR: node_defaults_with_some_metadata_default
    // What if there is default metadata for all rows?
    let defaults = DefaultsWithMetadata {
        metadata: Some(NodeMetadata { value: 42 }),
        ..Default::default()
    };
    // ANCHOR_END: node_defaults_with_some_metadata_default

    // ANCHOR: node_defaults_with_some_metadata_default_add_first_row
    let n0 = tables
        .add_node_with_defaults(
            0.0,
            &DefaultsWithMetadata {
                metadata: Some(NodeMetadata { value: 2 * 42 }),
                ..defaults
            },
        )
        .unwrap();
    // ANCHOR_END: node_defaults_with_some_metadata_default_add_first_row
    assert_eq!(
        tables
            .nodes()
            .metadata::<NodeMetadata>(n0)
            .unwrap()
            .unwrap(),
        NodeMetadata { value: 2 * 42 }
    );

    // But now, we start to cause a problem:
    // If we don't clone here, our metadata type moves,
    // so our defaults are moved.
    // ANCHOR: node_defaults_with_some_metadata_default_add_second_row
    let n1 = tables
        .add_node_with_defaults(
            0.0,
            &DefaultsWithMetadata {
                population: 6.into(),
                ..defaults.clone()
            },
        )
        .unwrap();
    // ANCHOR_END: node_defaults_with_some_metadata_default_add_second_row
    assert_eq!(
        tables
            .nodes()
            .metadata::<NodeMetadata>(n1)
            .unwrap()
            .unwrap(),
        NodeMetadata { value: 42 }
    );

    // Now, we have a use-after-move error
    // if we hadn't cloned in the last step.
    // ANCHOR: node_defaults_with_some_metadata_default_add_third_row
    let n2 = tables
        .add_node_with_defaults(
            0.0,
            &DefaultsWithMetadata {
                individual: 7.into(),
                ..defaults
            },
        )
        .unwrap();
    // ANCHOR_END: node_defaults_with_some_metadata_default_add_third_row
    assert_eq!(
        tables
            .nodes()
            .metadata::<NodeMetadata>(n2)
            .unwrap()
            .unwrap(),
        NodeMetadata { value: 42 }
    );
}