Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/numagic/lumos

scalable accelerated optimal control
https://github.com/numagic/lumos

Last synced: 3 months ago
JSON representation

scalable accelerated optimal control

Awesome Lists containing this project

README

        


drawing

# lumos: scalable optimal control for dynamical systems

[![CI with docker](https://github.com/numagic/lumos/actions/workflows/test_with_docker_image.yml/badge.svg)](https://github.com/numagic/lumos/actions/workflows/test_with_docker_image.yml)
[![CI with conda](https://github.com/numagic/lumos/actions/workflows/test_with_conda.yml/badge.svg)](https://github.com/numagic/lumos/actions/workflows/test_with_conda.yml)
[![PyPI version](https://badge.fury.io/py/numagic-lumos.svg)](https://badge.fury.io/py/numagic-lumos)
[![Downloads](https://pepy.tech/badge/numagic-lumos)](https://pepy.tech/project/numagic-lumos)

[**Benchmark on main**](https://numagic.github.io/lumos/dev/bench/)
| [**Benchmark on features**](https://numagic.github.io/lumos/dev/bench-features/)

[**Optimal Control Example**](#optimal-control-example)
| [**Quickstart**](#quickstart-colab-in-the-cloud)
| [**Environment setup**](#environment-setup)

## What is lumos?
[**lumos**](https://github.com/numagic/lumos) is a scalable numerical framework for the modelling and simulation of dynamical systems. Currently lumos focus on optimal control problems only, but the ideas it implements can be easily extended to other problems such as parameter inferences and state estimations.

One of the core philosophies of **lumos** is to leverage some fundamental technologies that enables the deep learning revolution: namely automatic differnetiation and large scale parallelization using accelerated hardware (GPU, TPU, etc).

**lumos** adopts a numpy-like syntax to write models, and currently has two built-in backends: [JAX](https://jax.readthedocs.io/en/latest/) and [Casadi](https://web.casadi.org). In addition to the built-in backends, **lumos** is also extensible and can solve optimal control problems with user-defined custom model bakends, provided that a few necessary APIs are exposed.

## Optimal Control Example
```python
import numpy as np
from typing import Any, Dict

import lumos.numpy as lnp
from lumos.models.base import StateSpaceModel, StateSpaceModelReturn, state_space_io
from lumos.optimal_control.config import (
BoundaryConditionConfig,
BoundConfig,
)
from lumos.optimal_control.scaled_mesh_ocp import ScaledMeshOCP

# Create a model
@state_space_io(states=("v", "x", "y"), inputs=("theta",), outputs=("theta",))
class TimeModel(StateSpaceModel):
"""The Brachistochrone model formulatd in the time domain"""

def __init__(
self, model_config: Dict[str, Any] = {}, params: Dict[str, Any] = {},
):
super().__init__(model_config=model_config, params=params)

def forward(
self,
states: Dict[str, float],
inputs: Dict[str, float],
mesh: float = 0.0, # time invariant model
) -> StateSpaceModelReturn:
params = self._params
theta = inputs["theta"]
v = states["v"]
v_dot = -params["gravity"] * lnp.sin(theta)

dx_dt = lnp.cos(theta) * v
dy_dt = lnp.sin(theta) * v

# Assemble result
states_dot = self.make_dict(group="states_dot", v=v_dot, x=dx_dt, y=dy_dt,)
outputs = self.make_dict(group="outputs", theta=theta)
return StateSpaceModelReturn(states_dot=states_dot, outputs=outputs)

@classmethod
def get_default_params(self) -> Dict[str, Any]:
return {"gravity": 9.81}

# Set up the model and the problem
model = TimeModel()
sim_config = ScaledMeshOCP.get_sim_config(
boundary_conditions=(
BoundaryConditionConfig(0, "states", "x", 0.0),
BoundaryConditionConfig(0, "states", "y", 0.0),
BoundaryConditionConfig(0, "states", "v", 0.0),
BoundaryConditionConfig(-1, "states", "x", 1.0),
BoundaryConditionConfig(-1, "states", "y", -0.6),
),
bounds=(
BoundConfig("global", "mesh_scale", (0.01, 10.0)),
BoundConfig("inputs", "theta", (-np.pi / 2, np.pi / 2)),
),
num_intervals=49,
hessian_approximation="exact",
)
ocp = ScaledMeshOCP(model=model, sim_config=sim_config)

# Solve with a trivial initial guess
solution, info = ocp.solve(
init_guess=np.zeros(ocp.num_dec), max_iter=200, print_level=4,
)
print(f"maneuveur time: {ocp.objective(solution):.3f} seconds")

```

### Contents
* [Optimal Control Example](#optimal-control-example)
* [Quickstart: Colab in the Cloud](#quickstart-colab-in-the-cloud)
* [Envirohnment setup](#environment-setup)

## Quickstart: Colab in the Cloud
- [Brachistochrone notebook](https://github.com/numagic/lumos/blob/main/tutorials/colab/Brachistochrone.ipynb) of the Brachistochrone example with the end-to-end implementation with and without automatic differentiation
- [Laptime simulation notebook](https://github.com/numagic/lumos/blob/main/tutorials/colab/Laptime_simulation.ipynb) for optimal laptime around a race track, using GPUs.

## Environment setup
**lumos** It is recommened to develop with lumos via two ways: using [**conda**](#setting-up-with-conda) or using [**docker environment**](#setting-up-with-docker), with the latter preferred, especially for developers as.

At the moment, **lumos** is tested and supports Python 3.7, 3.8 and 3.9.

### Setting up with conda
1) setting up the dependencies
```sh
conda create -n lumos python=3.9
conda env update -n lumos -f environment.yml
conda activate lumos
```
2) Install **lumos** from test.pypi (temporarily)
```sh
pip install -i https://test.pypi.org/simple/ numagic-lumos==0.0.5a0
```

2) Alternatively, install **lumos** from source
```sh
python3 -m pip install .
```

3) Test if it works
```sh
python3 examples/drone_example.py
```

### Setting up with docker

1) you can directy build the local dev container with:
```sh
docker-compose build --build-arg UID=$(id -u) --build-arg GID=$(id -g)
```

2) After the docker container is build, start it with
```sh
docker-compose up -d
```
The container should now be running in the background.

3) [Optional] download and install VSCode, and then install the 'remote containers' extension.

4) once VSCode with the extensions are installed, open VSCode, and open the folder where the repo is. It should automatically recognize the container configuration file, and asks if you want to open it in an conatiner, click 'Reopen in Container'


drawing

5) Once the VSCode enter the container, it will set up the other extensions and adopt settinsg defined in .devcontainer.json. After it installs the extensions, it might ask you to give permission to reload the conatiner to activate extensions like pylance language server, simply click yes.

6) at this point, you should be able to run the examples, in a conatiner, try:
```sh
(lumos) @lumos_dev:~/numagic/lumos$ python3 examples/drone_example.py
```
and the simulations should run and in the end you should get something like:
```sh
Number of objective function evaluations = 225
Number of objective gradient evaluations = 67
Number of equality constraint evaluations = 225
Number of inequality constraint evaluations = 0
Number of equality constraint Jacobian evaluations = 67
Number of inequality constraint Jacobian evaluations = 0
Number of Lagrangian Hessian evaluations = 66
Total seconds in IPOPT = 3.247

EXIT: Optimal Solution Found.
INFO:__main__:Maneuver time 1.300 sec
INFO:__main__:Final theta 6.28 rad
INFO:__main__:Final sin(theta) -0.00
```

7) [FIXME] currently the python extension setting for vscode extensions (which does not affect running the code, but affects hwo the VSCode extensions work) does not work automatically using the setting in .devcontainer.json. Therefore one must manually set the interpreter for tools such as linting with flake8, and autoformatting with black to work.
6.1) cmd + shift + p -> type 'select interpreter', and choose 'Python: select interpreter' when the option shows up
6.2) the lumos conda env should show up as an option (something like: Python 3.9.10 ('lumos': conda)). Choose this one.
6.3) Then linting and autoformatting should work.

#### Build the base images locally [Optional]
The aforementinoed process will download the base docker images from the public github repo, alternatively one could build the images locally:
```sh
$ docker-compose --env-file .env -f docker/docker-compose-build.yml build
```

#### Building GPU container [Optional]
Note: at the moment we switched the base image OS base to a smaller size linux base, to rebuild the GPU image, one needs to:
1) change to a GPU base image with cuda, eg: nvidia/cuda11.3.0-devel-ubuntu18.04
2) change the jaxlib version to a corresponding one with compatible cuda version. (sometimes one might also need to change the jax version because of jax and jaxlib version compatibility)

then to start the GPU containers, use multiple docker-compose see [here](https://docs.docker.com/compose/extends/)
```sh
docker-compose -f docker-compose.yml -f docker-compose.gpu.yml up -d
```

### Environment setup on Mac with M1 chip (Experimental)
It is possible to set up the environment on Mac with M1 chip using both the conda and the docker approach described earlier, but both will require some customization steps to tackle general problems that arise from library/OS/chip compatibilty.

From the speed perspetive, using conda directly seems to have a significant advantage, probably because it can better utilize the power of the M1 chip than via docker with emulation.

#### Setting up with Conda on M1 Mac
It follows the general approach outlined [above](#setting-up-with-conda), but two main changes:
1) install casadi with conda-forge instead of pip

Modify the environment.yml to remove cyipopt (we'll install it from source later), and move casadi from pip dependency to conda dependency. Add jax and jaxlib to conda dependencies, as shown below:
```yaml
channels:
- conda-forge
- defaults
dependencies:
- casadi
- jax
- jaxlib
- pip
- pip:
- casadi
- pyarrow
- pandas
```

And then create and update the conda environment as [before](#setting-up-with-conda)
```sh
conda create -n lumos_m1 python=3.9
conda env update --file environment.yml
```

Check if jax and casadi are both working (in a python terminal)
```python3
import jax.numpy as jnp
import numpy as np
import casadi as cas
aa = np.random.randn(10)

jnp.dot(aa, aa)
cas.dot(aa, aa)
```

2) install cyipopt from source

Clone the cyipopt repo, and move into repo directory:
```sh
git clone https://github.com/mechmotum/cyipopt.git
cd cyipopt
```

Check out a release tag (optional)
```sh
git checkout tags/v1.1.0
```

Install the dependencies required (use [this](https://github.com/mechmotum/cyipopt/blob/5e371bebb85a6f9ce0a53ebdd78daf9c696e4e84/.github/workflows/test.yml#L29) as a reference)

```sh
conda install -q -y lapack "libblas=*=*netlib" cython>=0.26 "ipopt=3.14" numpy>=1.15 pkg-config>=0.29.2 setuptools>=39.0
```

Install from source
```sh
python setup.py install
python -m pip install .
```

Check if cyipopt is working correctly
```sh
pip instal pytest
pytest
```

Test if all components are all working correctly (using lumos examples)
```sh
cd ..
python examples/laptime_simulation_example.py
```

#### Setting up with Docker on M1 Mac
It is also possible to set up the dev enviornment using the same dev conatiner, but we would need to replace the jax dependencies in the container as they were built with AVX which is not supported for docker on M1 (which uses Rosetta emulation)

So inside the container, one needs to replace `jax` and `jaxlib` with versions that are built WITHOUT AVX, see [here]([url](https://github.com/google/jax/issues/5501#issuecomment-1032891169))
```sh
conda install -c conda-forge jaxlib
conda install -c conda-forge jax
```
And then all should just work as before.

If one wants to build the corresponding container locally, this is equivalent to modifying the `enviorenment.yml` to:

```yaml
channels:
- conda-forge
- defaults
dependencies:
- cyipopt
- jax
- jaxlib
- pip
- pip:
- casadi
- pyarrow
- pandas
```

# Stargazers over time

[![Stargazers over time](https://starchart.cc/numagic/lumos.svg)](https://starchart.cc/numagic/lumos)