use std::collections::HashMap;
use std::iter::Iterator;

use nom::bytes::complete::tag;
use nom::character::complete::alphanumeric1;
use nom::combinator::iterator;
use nom::sequence::{separated_pair, terminated};
use nom::IResult;

fn main() {
  let mut data = "abcabcabcabc";

  fn parser(i: &str) -> IResult<&str, &str> {
    tag("abc")(i)
  }

  // `from_fn` (available from Rust 1.34) can create an iterator
  // from a closure
  let it = std::iter::from_fn(move || {
    match parser(data) {
      // when successful, a nom parser returns a tuple of
      // the remaining input and the output value.
      // So we replace the captured input data with the
      // remaining input, to be parsed on the next call
      Ok((i, o)) => {
        data = i;
        Some(o)
      }
      _ => None,
    }
  });

  for value in it {
    println!("parser returned: {}", value);
  }

  println!("\n********************\n");

  let data = "abcabcabcabc";

  // if `from_fn` is not available, it is possible to fold
  // over an iterator of functions
  let res =
    std::iter::repeat(parser)
      .take(3)
      .try_fold((data, Vec::new()), |(data, mut acc), parser| {
        parser(data).map(|(i, o)| {
          acc.push(o);
          (i, acc)
        })
      });

  // will print "parser iterator returned: Ok(("abc", ["abc", "abc", "abc"]))"
  println!("\nparser iterator returned: {:?}", res);

  println!("\n********************\n");

  let data = "key1:value1,key2:value2,key3:value3,;";

  // `nom::combinator::iterator` will return an iterator
  // producing the parsed values. Compared to the previous
  // solutions:
  // - we can work with a normal iterator like `from_fn`
  // - we can get the remaining input afterwards, like with the `try_fold` trick
  let mut nom_it = iterator(
    data,
    terminated(
      separated_pair(alphanumeric1, tag(":"), alphanumeric1),
      tag(","),
    ),
  );

  let res = nom_it
    .map(|(k, v)| (k.to_uppercase(), v))
    .collect::<HashMap<_, _>>();

  let parser_result: IResult<_, _> = nom_it.finish();
  let (remaining_input, ()) = parser_result.unwrap();

  // will print "iterator returned {"key1": "value1", "key3": "value3", "key2": "value2"}, remaining input is ';'"
  println!(
    "iterator returned {:?}, remaining input is '{}'",
    res, remaining_input
  );
}