/*
 * SPDX-FileCopyrightText: 2023 Inria
 * SPDX-FileCopyrightText: 2023 Sebastiano Vigna
 *
 * SPDX-License-Identifier: Apache-2.0 OR LGPL-2.1-or-later
 */

use super::*;
use anyhow::Result;
use dsi_bitstream::prelude::*;

/// Fast iterator over the offsets and degrees of a [`BvGraph`].
///
/// This iterator is faster than scanning the graph. In particular, it can be
/// used to build the offsets of a graph or to enumerate the graph degrees when
/// the offsets are not available.
#[derive(Debug, Clone)]
pub struct OffsetDegIter<D: Decode> {
    number_of_nodes: usize,
    compression_window: usize,
    min_interval_length: usize,
    node_id: usize,
    decoder: D,
    backrefs: Vec<usize>,
}

impl<D: Decode + BitSeek> OffsetDegIter<D> {
    /// Get the current bit offset in the bitstream.
    pub fn get_pos(&mut self) -> u64 {
        self.decoder.bit_pos().unwrap()
    }
}

impl<D: Decode + BitSeek> Iterator for OffsetDegIter<D> {
    type Item = (u64, usize);
    fn next(&mut self) -> Option<(u64, usize)> {
        if self.node_id >= self.number_of_nodes {
            return None;
        }
        let offset = self.get_pos();
        Some((offset, self.next_degree().unwrap()))
    }
}

impl<D: Decode + BitSeek> ExactSizeIterator for OffsetDegIter<D> {
    fn len(&self) -> usize {
        self.number_of_nodes - self.node_id
    }
}

impl<D: Decode> OffsetDegIter<D> {
    /// Creates a new iterator over the degrees of the graph.
    pub fn new(
        decoder: D,
        number_of_nodes: usize,
        compression_window: usize,
        min_interval_length: usize,
    ) -> Self {
        Self {
            number_of_nodes,
            compression_window,
            min_interval_length,
            node_id: 0,
            decoder,
            backrefs: vec![0; compression_window],
        }
    }

    /// Creates a new iterator over the degrees of the graph at a given node
    pub fn new_from(
        decoder: D,
        number_of_nodes: usize,
        compression_window: usize,
        min_interval_length: usize,
        node_id: usize,
        backrefs: Vec<usize>,
    ) -> Self {
        debug_assert_eq!(backrefs.len(), compression_window);
        Self {
            number_of_nodes,
            compression_window,
            min_interval_length,
            node_id,
            decoder,
            backrefs,
        }
    }

    /// Get the number of nodes in the graph
    #[inline(always)]
    pub fn num_nodes(&self) -> usize {
        self.number_of_nodes
    }

    /// Returns a reference to the decoder so it can be inspected during iteration.
    pub fn get_decoder(&self) -> &D {
        &self.decoder
    }

    /// Convert the decoder to another one.
    pub fn map_decoder<D2: Decode, F: FnOnce(D) -> D2>(self, f: F) -> OffsetDegIter<D2> {
        OffsetDegIter {
            decoder: f(self.decoder),
            backrefs: self.backrefs,
            node_id: self.node_id,
            min_interval_length: self.min_interval_length,
            compression_window: self.compression_window,
            number_of_nodes: self.number_of_nodes,
        }
    }

    #[inline(always)]
    /// Manually get the next degree, this is what the iterator calls internally
    /// but it calls `.unwrap()` on it because the trait Graph doesn't allows
    /// errors.
    pub fn next_degree(&mut self) -> Result<usize> {
        let degree = self.decoder.read_outdegree() as usize;
        // no edges, we are done!
        if degree == 0 {
            if self.compression_window != 0 {
                self.backrefs[self.node_id % self.compression_window] = degree;
            }
            self.node_id += 1;
            return Ok(degree);
        }

        let mut nodes_left_to_decode = degree;

        // read the reference offset
        let ref_delta = if self.compression_window != 0 {
            self.decoder.read_reference_offset() as usize
        } else {
            0
        };
        // if we copy nodes from a previous one
        if ref_delta != 0 {
            // compute the node id of the reference
            let reference_node_id = self.node_id - ref_delta;
            // retrieve the data
            let ref_degree = self.backrefs[reference_node_id % self.compression_window];
            // get the info on which destinations to copy
            let number_of_blocks = self.decoder.read_block_count() as usize;

            // no blocks, we copy everything
            if number_of_blocks == 0 {
                nodes_left_to_decode -= ref_degree;
            } else {
                // otherwise we copy only the blocks of even index

                // the first block could be zero
                let mut idx = self.decoder.read_block() as usize;
                nodes_left_to_decode -= idx;

                // while the other can't
                for block_id in 1..number_of_blocks {
                    let block = self.decoder.read_block() as usize;
                    let end = idx + block + 1;
                    if block_id % 2 == 0 {
                        nodes_left_to_decode -= block + 1;
                    }
                    idx = end;
                }
                if number_of_blocks & 1 == 0 {
                    nodes_left_to_decode -= ref_degree - idx;
                }
            }
        };

        // if we still have to read nodes
        if nodes_left_to_decode != 0 && self.min_interval_length != 0 {
            // read the number of intervals
            let number_of_intervals = self.decoder.read_interval_count() as usize;
            if number_of_intervals != 0 {
                // pre-allocate with capacity for efficiency
                let _ = self.decoder.read_interval_start();
                let mut delta = self.decoder.read_interval_len() as usize;
                delta += self.min_interval_length;
                // save the first interval
                nodes_left_to_decode -= delta;
                // decode the intervals
                for _ in 1..number_of_intervals {
                    let _ = self.decoder.read_interval_start();
                    delta = self.decoder.read_interval_len() as usize;
                    delta += self.min_interval_length;

                    nodes_left_to_decode -= delta;
                }
            }
        }

        // decode the extra nodes if needed
        if nodes_left_to_decode != 0 {
            // pre-allocate with capacity for efficiency
            let _ = self.decoder.read_first_residual();
            for _ in 1..nodes_left_to_decode {
                let _ = self.decoder.read_residual();
            }
        }
        if self.compression_window != 0 {
            self.backrefs[self.node_id % self.compression_window] = degree;
        }
        self.node_id += 1;
        Ok(degree)
    }
}
