> ## Documentation Index
> Fetch the complete documentation index at: https://mathematicalcompany.mintlify.site/llms.txt
> Use this file to discover all available pages before exploring further.

# Entropy Pooling

> Meucci's entropy pooling for blending subjective views with market-implied probabilities. Rust-native convex optimization with Python pipeline integration.

<Warning>
  **Ultra Feature.** Requires an Ultra subscription. [Get started at api.mathematicalcompany.com](https://api.mathematicalcompany.com)
</Warning>

<Tip>
  **What is this?** You have a view ('I think candidate X has a 70% chance of winning') and historical scenario data. Entropy pooling finds the probability distribution closest to your prior that satisfies your views. Unlike Black-Litterman, it works with any distribution, supports inequality views ('at least 60%'), and handles non-linear constraints. Use it to systematically blend subjective views with data-driven priors.
</Tip>

# Entropy Pooling

Entropy pooling (Meucci, 2008) is a fully general framework for blending subjective views with a prior distribution over scenarios. Unlike Black-Litterman, which is limited to normal distributions and linear views, entropy pooling works with arbitrary scenario sets and supports inequality constraints, conditional views, and non-linear payoffs. Horizon implements the dual optimization in Rust for fast convergence.

<CardGroup cols={2}>
  <Card title="View Blending" icon="scale-balanced">
    `hz.entropy_pool()` tilts a prior distribution to satisfy your views while staying as close as possible to the prior (minimum relative entropy).
  </Card>

  <Card title="Flexible Views" icon="sliders">
    Express equality and inequality constraints on means, variances, correlations, or any linear function of the scenarios.
  </Card>

  <Card title="Posterior Analytics" icon="chart-pie">
    `hz.posterior_mean()` and `hz.posterior_covariance()` extract portfolio-ready moments from the reweighted distribution.
  </Card>

  <Card title="Pipeline Integration" icon="plug">
    `hz.entropy_pooling()` pipeline function updates posterior weights each cycle as views evolve.
  </Card>
</CardGroup>

***

## hz.entropy\_pool

Compute posterior scenario probabilities that satisfy a set of view constraints while minimizing the Kullback-Leibler divergence from the prior.

```python theme={null}
import horizon as hz

# 5 scenarios, 3 markets
scenarios = [
    [0.10, -0.05,  0.02],
    [0.05,  0.03, -0.01],
    [-0.02, 0.08,  0.04],
    [0.03, -0.02,  0.06],
    [-0.08, 0.01,  0.03],
]

# Equal prior (uniform)
prior_probs = [0.2, 0.2, 0.2, 0.2, 0.2]

# Views: "expected return of market 0 is at least 3%"
views = [
    hz.ViewConstraint(
        coefficients=[1.0, 0.0, 0.0],  # select market 0
        bound=0.03,
        equality=False,  # inequality: E[r_0] >= 0.03
    ),
]

result = hz.entropy_pool(prior_probs, scenarios, views)
print(result.posterior_probs)   # reweighted scenario probabilities
print(result.relative_entropy)  # KL divergence from prior
print(result.converged)         # True if optimization converged
```

| Parameter     | Type                   | Description                                              |
| ------------- | ---------------------- | -------------------------------------------------------- |
| `prior_probs` | `list[float]`          | Prior scenario probabilities, length S (must sum to 1.0) |
| `scenarios`   | `list[list[float]]`    | S x N scenario matrix (S scenarios, N markets)           |
| `views`       | `list[ViewConstraint]` | List of view constraints to impose on the posterior      |

### EntropyPoolResult Type

| Field              | Type          | Description                                                |   |                                                                  |
| ------------------ | ------------- | ---------------------------------------------------------- | - | ---------------------------------------------------------------- |
| `posterior_probs`  | `list[float]` | Posterior scenario probabilities, length S, summing to 1.0 |   |                                                                  |
| `relative_entropy` | `float`       | KL divergence D(posterior                                  |   | prior). Lower means the views are more compatible with the prior |
| `converged`        | `bool`        | Whether the dual optimization converged within tolerance   |   |                                                                  |
| `iterations`       | `int`         | Number of iterations taken                                 |   |                                                                  |

### ViewConstraint Type

| Field          | Type          | Description                                                                                             |
| -------------- | ------------- | ------------------------------------------------------------------------------------------------------- |
| `coefficients` | `list[float]` | Linear combination weights, length N. The constraint applies to sum\_j coefficients\[j] \* scenario\[j] |
| `bound`        | `float`       | The target value for the view                                                                           |
| `equality`     | `bool`        | `True` for equality constraint (E\[g(X)] = bound), `False` for inequality (E\[g(X)] >= bound)           |

***

## Expressing Views

### Mean View (Equality)

"I believe the expected return of market 1 is exactly 5%."

```python theme={null}
view = hz.ViewConstraint(
    coefficients=[0.0, 1.0, 0.0],
    bound=0.05,
    equality=True,
)
```

### Mean View (Inequality)

"I believe market 2 will return at least 2%."

```python theme={null}
view = hz.ViewConstraint(
    coefficients=[0.0, 0.0, 1.0],
    bound=0.02,
    equality=False,
)
```

### Relative View

"Market 0 will outperform market 1 by at least 3%."

```python theme={null}
view = hz.ViewConstraint(
    coefficients=[1.0, -1.0, 0.0],
    bound=0.03,
    equality=False,
)
```

### Multiple Views

```python theme={null}
views = [
    hz.ViewConstraint([1.0, 0.0, 0.0], bound=0.04, equality=False),
    hz.ViewConstraint([0.0, 1.0, -1.0], bound=0.01, equality=True),
]
result = hz.entropy_pool(prior_probs, scenarios, views)
```

***

## Posterior Moments

After computing posterior probabilities, extract the mean and covariance for portfolio construction.

### hz.posterior\_mean

```python theme={null}
import horizon as hz

mean = hz.posterior_mean(result.posterior_probs, scenarios)
print(mean)  # [0.042, 0.031, 0.025] - posterior expected returns
```

| Parameter   | Type                | Description                                  |
| ----------- | ------------------- | -------------------------------------------- |
| `probs`     | `list[float]`       | Scenario probabilities (posterior), length S |
| `scenarios` | `list[list[float]]` | S x N scenario matrix                        |

Returns `list[float]`: posterior mean vector, length N.

### hz.posterior\_covariance

```python theme={null}
cov = hz.posterior_covariance(result.posterior_probs, scenarios)
print(cov)  # 3x3 posterior covariance matrix
```

| Parameter   | Type                | Description                                  |
| ----------- | ------------------- | -------------------------------------------- |
| `probs`     | `list[float]`       | Scenario probabilities (posterior), length S |
| `scenarios` | `list[list[float]]` | S x N scenario matrix                        |

Returns `list[list[float]]`: posterior N x N covariance matrix.

***

## Pipeline Integration

The `hz.entropy_pooling()` pipeline function recomputes posterior probabilities each cycle as your model views evolve, injecting the result into `ctx.params["entropy_pool"]`.

```python theme={null}
import horizon as hz

def view_generator(ctx):
    """Generate views from your model each cycle."""
    price = ctx.feeds["poly"].price
    views = []
    if price < 0.50:
        # Bullish view: expected return >= 5%
        views.append(hz.ViewConstraint([1.0], bound=0.05, equality=False))
    return views

def model(ctx):
    ep = ctx.params.get("entropy_pool")
    if ep is None or not ep.converged:
        return []

    posterior_mean = hz.posterior_mean(ep.posterior_probs, ctx.params["scenarios"])
    fair = 0.50 + posterior_mean[0]
    return hz.quotes(fair=fair, spread=0.04, size=10)

hz.run(
    name="entropy-pooling-mm",
    markets=["election"],
    pipeline=[
        hz.entropy_pooling(
            feed="poly",
            scenario_lookback=100,
            view_fn=view_generator,
        ),
        model,
    ],
    feeds={"poly": hz.PolymarketBook(token_id="0x123...")},
    interval=5.0,
)
```

### Parameters

| Parameter           | Type       | Default  | Description                                                              |
| ------------------- | ---------- | -------- | ------------------------------------------------------------------------ |
| `feed`              | `str`      | required | Feed name to build scenarios from                                        |
| `scenario_lookback` | `int`      | `100`    | Number of historical observations to use as scenarios                    |
| `view_fn`           | `callable` | required | Function `(ctx) -> list[ViewConstraint]` that generates views each cycle |

***

## Mathematical Background

<AccordionGroup>
  <Accordion title="Entropy Pooling Algorithm">
    Given S scenarios with prior probabilities p = (p\_1, ..., p\_S) and K view constraints of the form E\_q\[g\_k(X)] = b\_k (or >= b\_k), entropy pooling finds the posterior q that minimizes:

    D(q || p) = sum\_s q\_s \* ln(q\_s / p\_s)

    subject to sum\_s q\_s = 1 and the view constraints. The dual problem is unconstrained and convex, solved via Newton's method. The posterior has an exponential-family form:

    q\_s proportional to p\_s \* exp(sum\_k lambda\_k \* g\_k(x\_s))

    where lambda\_k are the dual variables (Lagrange multipliers).
  </Accordion>

  <Accordion title="Comparison with Black-Litterman">
    Black-Litterman is a special case of entropy pooling where: (a) the prior is normal, (b) views are linear equality constraints on expected returns, and (c) the posterior is also normal. Entropy pooling generalizes this to arbitrary scenario distributions, inequality constraints, and non-Gaussian priors. This is important for prediction markets where return distributions are bounded (prices must stay in \[0, 1]) and heavy-tailed.
  </Accordion>

  <Accordion title="Relative Entropy Interpretation">
    The relative entropy D(q || p) measures the information cost of the views. Views that are highly inconsistent with the prior will produce a large D, indicating that the posterior has moved far from your baseline. Monitoring D over time can signal when your model views are becoming extreme or contradictory.
  </Accordion>
</AccordionGroup>

<Warning>
  Entropy pooling requires the number of scenarios S to be larger than the number of equality constraints. If you impose too many exact equality views relative to the scenario count, the problem becomes infeasible. Use inequality views when possible, and ensure S >> K.
</Warning>
