jellyflow-runtime 0.1.0

Headless store, rules, schema, profile, and change pipeline for Jellyflow.
Documentation
use std::collections::BTreeSet;

use crate::runtime::xyflow::callbacks::{ConnectionChange, EdgeConnection};
use jellyflow_core::core::{Edge, EdgeId};
use jellyflow_core::ops::GraphOp;

use super::removed_edges::visit_removed_edges;

pub(super) struct ConnectionChangeAccumulator {
    out: Vec<ConnectionChange>,
    removed_edges: BTreeSet<EdgeId>,
}

impl ConnectionChangeAccumulator {
    pub(super) fn new(op_count: usize) -> Self {
        let mut out = Vec::new();
        out.reserve(op_count.min(8));
        Self {
            out,
            removed_edges: BTreeSet::new(),
        }
    }

    pub(super) fn push_op(&mut self, op: &GraphOp) {
        if visit_removed_edges(op, |id, edge| self.push_disconnected_edge(id, edge)) {
            return;
        }

        match op {
            GraphOp::AddEdge { id, edge } => self.push_connected(*id, edge),
            GraphOp::SetEdgeEndpoints { id, from, to } => {
                self.out.push(ConnectionChange::Reconnected {
                    edge: *id,
                    from: *from,
                    to: *to,
                });
            }
            _ => {}
        }
    }

    fn push_connected(&mut self, id: EdgeId, edge: &Edge) {
        self.removed_edges.remove(&id);
        self.out
            .push(ConnectionChange::Connected(EdgeConnection::from_edge(
                id, edge,
            )));
    }

    fn push_disconnected_edge(&mut self, id: EdgeId, edge: &Edge) {
        if self.removed_edges.insert(id) {
            self.out
                .push(ConnectionChange::Disconnected(EdgeConnection::from_edge(
                    id, edge,
                )));
        }
    }

    pub(super) fn finish(self) -> Vec<ConnectionChange> {
        self.out
    }
}