# Factor Graph Project

A modular Python framework for factor graphs, belief propagation, and chemical reaction network compilation.

## Overview

This project implements:

1. **Factor Graph Construction** - Flexible specification of factor graphs with arbitrary variables and factors
2. **Belief Propagation** - Sum-product message passing inference (to be implemented)
3. **Poset Reductions** - Deformation retractions following Sergeant-Perthuis & Boitel
4. **CRN Compilation** - Convert factor graphs to chemical reaction networks following Napp & Adams
5. **Verification** - Ensure CRN steady states match BP fixed points

## Project Structure

```
factor_graph_project/
├── __init__.py
├── README.md
├── core/
│   ├── __init__.py
│   ├── factor_graph.py    
│   └── poset.py           
├── inference/
│   ├── __init__.py
│   └── belief_propagation.py   
├── reduction/
│   ├── __init__.py
│   └── graph_reduction.py      
├── crn/
│   ├── __init__.py
│   ├── species.py              
│   ├── reactions.py            
│   └── simulation.py           
├── examples/
│   ├── __init__.py
│   └── factor_graph_examples.py
└── tests/
    └── ...
```

## Quick Start

```python
from core import Variable, Factor, FactorGraph, build_factor_graph
import numpy as np

# Method 1: Step-by-step construction
fg = FactorGraph("MyGraph")

x1 = Variable("x1", [0, 1])      # Binary variable
x2 = Variable("x2", [0, 1, 2])   # Ternary variable

fg.add_variables(x1, x2)

# Unary factor on x1
f1 = Factor("f1", [x1], np.array([0.7, 0.3]))

# Pairwise factor on (x1, x2)
f2 = Factor("f2", [x1, x2], np.array([
    [1.0, 0.5, 0.2],  # x1=0
    [0.2, 0.5, 1.0]   # x1=1
]))

fg.add_factors(f1, f2)

# Method 2: Using convenience function
fg = build_factor_graph(
    variables_spec={
        "x1": [0, 1],
        "x2": [0, 1, 2],
    },
    factors_spec={
        "f1": (["x1"], np.array([0.7, 0.3])),
        "f2": (["x1", "x2"], np.array([[1.0, 0.5, 0.2], [0.2, 0.5, 1.0]])),
    },
    name="MyGraph"
)

# Compute exact marginals (exponential complexity)
for var in fg.variables:
    marginal = fg.compute_marginal_exact(var)
    print(f"P({var.name}) = {marginal}")
```

## References

1. **Napp & Adams (2013)**: "Message Passing Inference with Chemical Reaction Networks"
   - Compilation of sum-product BP to CRNs
   - Belief species representing messages
   - Equilibrium concentrations = marginal probabilities

2. **Sergeant-Perthuis & Boitel**: "Minima and Critical Points of the Bethe Free Energy Are Invariant Under Deformation Retractions of Factor Graphs"
   - Poset perspective on factor graphs
   - Linear/colinear point retractions
   - Preservation of Bethe free energy critical points