Metadata-Version: 2.3
Name: onsi-mc
Version: 0.1.2
Summary: A Python library for Monte Carlo N-dimensional integration
License: GNU GENERAL PUBLIC LICENSE
         Version 3, 29 June 2007
         
         Copyright (C) 2024  Ali Onsi <aonsi@alexu.edu.eg>
         
         mc - A Python library for Monte Carlo N-dimensional integration
         This program is free software: you can redistribute it and/or modify
         it under the terms of the GNU General Public License as published by
         the Free Software Foundation, either version 3 of the License, or
         (at your option) any later version.
         
         This program is distributed in the hope that it will be useful,
         but WITHOUT ANY WARRANTY; without even the implied warranty of
         MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
         GNU General Public License for more details.
         
         You should have received a copy of the GNU General Public License
         along with this program.  If not, see <https://www.gnu.org/licenses/>.
Keywords: monte-carlo,integration,numerical-integration,high-dimensional,scientific-computing,python-library
Author: Ali Onsi
Author-email: aonsi@alexu.edu.eg
Requires-Python: >=3.8
Classifier: License :: OSI Approved :: GNU General Public License v3 (GPLv3)
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Requires-Dist: matplotlib (>=3.0.0)
Requires-Dist: numba (>=0.56.0)
Requires-Dist: numpy (>=1.21.0)
Requires-Dist: tqdm (>=4.0.0)
Description-Content-Type: text/markdown

<!-- SPDX-FileCopyrightText: 2025 Ali Onsi -->
<!-- SPDX-License-Identifier: GPL-3.0-or-later -->

## Evaluating N-Dimensional Integration with Monte Carlo Methods
### Mathematical Basis
Another problem which can be solved using Monte Carlo's method is numerical integration. From the definition of the average of a function we have:

$$\langle f(x) \rangle = \frac{1}{b-a} \int_a^b f(x) dx$$

which can be rearranged as:

$$\int_a^b f(x) dx = (b-a) \langle f(x) \rangle$$

From this, we can conclude that with a large number of random points, we can approximate the value of the integral. If we have a function $f(x)$ and want to evaluate the integral from $a$ to $b$, we:

- Generate a large number of random points in the interval $[a,b]$ 
- Calculate the average value of $f(x)$ at these random points
- = Multiply this average by the length of the interval $(b-a)$

### Extension to Higher Dimensions 

This concept extends to 2D, 3D, and higher dimensional integrals:

For a 2D integral:

$$\langle f(x,y) \rangle = \frac{1}{b_x-a_x} , \frac{1}{b_y-a_y} , \int_{a_y}^{b_y} \int_{a_x}^{b_x} f(x,y) , dx , dy$$

Rearranged as:

$$\int_{a_y}^{b_y} \int_{a_x}^{b_x} f(x,y) , dx , dy = (b_x-a_x)(b_y-a_y) \langle f(x,y) \rangle$$

For a 3D integral:

$$\int_{a_z}^{b_z} \int_{a_y}^{b_y} \int_{a_x}^{b_x} f(x,y,z) , dx , dy , dz = (b_x-a_x)(b_y-a_y)(b_z-a_z) \langle f(x,y,z) \rangle$$

A general pattern emerges for $N$-dimensional integrals:

$$\int_{a_N}^{b_N} \cdots \int_{a_2}^{b_2} \int_{a_1}^{b_1} f(x_1, x_2, \ldots, x_N) , dx_1 , dx_2 , \ldots , dx_N = \left[\prod_{i=1}^N (b_i-a_i) \right] \cdot \langle f(x_1,x_2, \ldots, x_N) \rangle$$

What's impressive is that the difficulty of integrating a higher-order integral doesn't increase significantly with Monte Carlo methods, making it suitable for complex integrals.

## Installation

Install the package from PyPI:

```bash
pip install onsi-mc
```
### Quick Start
```
import numpy as np
from onsi_mc import ND

# Define a function to integrate
def func(x1, x2, x3):
    """3D integration function with vectorized support"""
    return np.exp(-x1**2) * np.sin(x2) * np.cos(x3)

# Create integrator with integration bounds
mc = ND(func, 0.0, 1.0, 0.0, np.pi, 0.0, np.pi/2)

# Run the integration
result = mc.run()

# Plot convergence analysis
mc.plot(save=True)
```
### API Reference
ND class: `ND(func, *args, n=50000, N=1000, convergence_threshold=0.1, use_numba=True)`
- `func`: Function to integrate. Must accept N arguments.
- `*args`: Integration bounds. Must be in pairs for each dimension. For example, for 3D: `a1, b1, a2, b2, a3, b3`.
- `n`: Number of random points to sample. Default is 50,000.
- `N`: Number of iterations for convergence analysis. Default is 1,000.
- `convergence_threshold`: Threshold for convergence analysis. Default is 0.5.
- `use_numba`: If True, uses Numba for JIT compilation. Default is True.

#### Methods
`run(verbose=True, progress_bar=True)`
This method runs the Monte Carlo integration and returns the result.
- `verbose`: If True, prints the result. Default is True.
- `progress_bar`: If True, shows a progress bar. Default is True.

`plot(show=True, save=False, filename=None)`
Visualizes convergence analysis.
- `show`: If True, displays the plot. Default is True.
- `save`: Saves plot to file. Default is False.
- `filename`:  Custom filename (default: "mc_convergence_n{n}_N{N}.png").


### Example Usage

```
from mc import ND
import numpy as np

# 1D integration example: ∫(x³ - 2x + 1)dx from 0 to 2
def f1d(x):
    return x**3 - 2*x + 1

# Create integrator with 0 to 2 limits
mc1 = ND(f1d, 0, 2)
result1 = mc1.run()

# 3D integration example
def f3d(x, y, z):
    return np.exp(-x**2) * np.sin(y) * np.cos(z)

# Create integrator with limits [0,1] × [0,π] × [0,π/2]
mc3 = ND(
    f3d, 
    0.0, 1.0,   # x limits
    0.0, np.pi, # y limits
    0.0, np.pi/2 # z limits
)

# Run with default settings
result3 = mc3.run()

# Plot convergence analysis
mc3.plot(save=True)
```

### Expected Output
When running the integration, you'll see one of two outcomes:
1. Successful Convergence
If the integration converges properly (standard deviation difference is below threshold):
```
Integrating: 100%|██████████| 1000/1000 [00:02<00:00, 396.60it/s]
Estimated integral: 0.60653211
```
The function returns this value, which you can use in subsequent calculations.
2. Non-Convergence Warning
If the integration fails to converge (standard deviation difference exceeds threshold):
```
Integrating: 100%|██████████| 1000/1000 [00:03<00:00, 327.42it/s]
Warning: Possible non-convergence (Δstd = 0.1237 > threshold = 0.1000)
Estimated integral: 0.60912458
```

In this case, the function returns None to indicate possible unreliability, though you can still access the estimated value as shown in the output message.

You can visualize the convergence behavior using the plot() method to determine whether increasing N or adjusting other parameters might help achieve convergence.

