liwe 0.1.10

IWE core library
Documentation
use indoc::indoc;
use liwe::graph::Graph;
use liwe::model::config::MarkdownOptions;
use liwe::query::execute;
use liwe::query::prelude::{delete, eq, filter};
use liwe::query::{DeleteOp, Filter, Outcome};
use liwe::state::from_indoc;

fn assert_delete(docs: &str, op: DeleteOp, expected: &[&str]) {
    let graph = Graph::import(&from_indoc(docs), MarkdownOptions::default(), None);
    match execute(&delete(op), &graph) {
        Outcome::Delete { removed } => {
            let actual: Vec<String> = removed.iter().map(|k| k.to_string()).collect();
            let expected: Vec<String> = expected.iter().map(|s| s.to_string()).collect();
            assert_eq!(actual, expected);
        }
        other => panic!("expected Delete, got {:?}", other),
    }
}

#[test]
fn delete_removes_matching() {
    assert_delete(
        indoc! {"
            ---
            status: archived
            ---
            # A
            _
            ---
            status: draft
            ---
            # B
            _
            ---
            status: archived
            ---
            # C
        "},
        filter(eq("status", "archived")),
        &["1", "3"],
    );
}

#[test]
fn delete_with_empty_filter_matches_all() {
    assert_delete(
        indoc! {"
            # A
            _
            # B
        "},
        filter(Filter::all()),
        &["1", "2"],
    );
}

#[test]
fn delete_respects_limit() {
    assert_delete(
        indoc! {"
            ---
            status: archived
            ---
            # A
            _
            ---
            status: archived
            ---
            # B
            _
            ---
            status: archived
            ---
            # C
        "},
        filter::<DeleteOp>(eq("status", "archived")).limit(2),
        &["1", "2"],
    );
}