Skip to main content

Python-interface Particle Sequential Convex Programming Model Predictive Control (SCP PMPC) interface.

Project description

pmpc

Python-interface Particle Sequential Convex Programming Model Predictive Control (SCP PMPC) interface.

This is non-linear dynamics finite horizon MPC solver with consensus optimization capability, support for arbitrary constraints and arbitrary cost.

Table of Contents

Quick Start

Take a look at examples/simple_demo.ipynb.

Installation

We offer precompiled binaries for Linux and MacOS (x86_64 and aarch64) via PyPI.

$ pip install pmpc

Even precompiled binaries have some startup time, so time only the second run of the solver.

Installation from Source

Installation can be done by cloning this repository and issuing pip install .

$ git clone https://github.com/StanfordASL/pmpc.git
$ cd pmpc
$ pip install .

We require that

  • you must have julia in your system PATH
  • Julia version is 1.6+

We highly recommend that

  • obtain a dynamically linked Python
    • which will make the julia Python interface startup much faster
  • obtain Julia 1.9
    • this version introduces a vastly faster time-to-first-run via better compilation utilities

Further subsections explain some optional installation steps.

(Optional) Obtaining (a dynamically linked version of) Python

The Python module uses pyjulia to be able to call Julia from Python which has a limitation in that it works much better (faster startup time) with a Python version which is not statically linked to libpython. A great workaround, obtaining a python version that is not statically linked can be easily done via pyenv.

Instructions on how to build a python version dynamically linked to libpython can be found

  • in pyjulia documentation here
  • or in pyenv documentation here

With a working version of pyenv, an example might be

$ env PYTHON_CONFIGURE_OPTS="--enable-shared" pyenv install 3.9.13
$ pyenv virtualenv 3.9.13 {your venv name} 
$ pyenv activate {your venv name}

Compilation Times and a Persistent Solver Process

A large downside of using a Julia for the core solver is that compilation needs to occur every time a new process is launched. This is exacerbated by pyjulia which does not work well with not-dynamically linked python interpreters (most commonly used).

To overcome compilation times, we can start a solver process once and the library can use that solver process repeatedly, even through script restarts.

Use

$ python3 -m pmpc.remote

to start a persistent solver process.

Next, from your script

>>> from pmpc.remote import solve # instead of `from pmpc import solve`

Obtaining Julia

You can download Julia, the programming language and interpreter, from here.

Make sure Julia is in your PATH.

Basic Usage

The solver is capable of MPC consensus optimization for several system instantiations. For the basic usage, we'll focus on a single system MPC.

A basic MPC problem is defined using the dynamics and a quadratic cost

Defining dynamics

  • x0 the initial state of shape

where

np.shape(x0) == (xdim,)
  • f, fx, fu = f_fx_fu_fn(xt, ut) an affine dynamics linearization, such that $$x^{(i+1)} \approx f^{(i)} + f_x^{(i)} (x^{(i)} - \tilde{x}^{(i)}) + f_u^{(i)} (u^{(i)} - \tilde{u}^{(i)}) $$ where
np.shape(xt) == (N, xdim)
np.shape(ut) == (N, udim)
np.shape(f) == (N, xdim)
np.shape(fx) == (N, xdim, xdim)
np.shape(fu) == (N, xdim, udim)

Defining Cost

  • X_ref, Q a reference and quadratic weight matrix for state cost
  • U_ref, R a reference and quadratic weight matrix for control cost

The cost is given as

$$J = \sum_{i=0}^N \frac{1}{2} (x^{(i+1)} - x_\text{ref}^{(i+1)}) Q^{(i)} (x^{(i+1)} - x_\text{ref}^{(i+1)}) + \frac{1}{2} (u^{(i)} - u_\text{ref}^{(i)}) R^{(i)} (u^{(i)} - u_\text{ref}^{(i)})$$

Note: Initial state, x0, is assumed constant and thus does not feature in the cost.

Note: When handling controls, we'll always have np.shape(U) == (N, udim)

Note: When handling states, we'll have either np.shape(X) == (N + 1, xdim) with x0 included at the beginning or np.shape(X) == (N, xdim) with x0 NOT INCLUDED. X[:, -1] always refers to the state N, whereas U[:, -1] always refers to control N - 1.

Thus, an example call would be

>>> import pmpc
>>> X, U, debug = pmpc.solve(f_fx_fu_fn, Q, R, x0, X_ref, U_ref)
>>> help(pmpc.solve)

Take a look at

  • tests/simple.py for simple usage
  • tests/dubins_car.py for defining dynamics

solve Method Arguments Glossary

Solver Hyperparameters

The solver has two scalar hyperparamters, the dynamics linearization deviation penalty for states and controls

$$J_\text{deviation} = \sum_{i=0}^N \frac{1}{2} \rho_x (x^{(i+1)} - x_\text{prev}^{(i+1)})^T (x^{(i+1)} - x_\text{prev}^{(i+1)}) + \rho_u (u^{(i)} - u_\text{prev}^{(i)})^T (u^{(i)} - u_\text{prev}^{(i)})$$

  • reg_x - state deviation in-between SCP iterations regularization
  • reg_u - control deviation in-between SCP iterations regularization

Higher values will slow evolution between SCP iterations and will require more SCP iterations to converge to a solution, but will avoid instability in the solution if the dynamics are not sufficiently smooth.

Solver Settings

  • verbose - whether to print iteration status (user-facing)
  • debug - whether to print very low-level debugging information (developer-facing)
  • max_it - maximum number of SCP iterations to perform (can be fewer if tolerance met earlier)
  • time_limit - the time limit in seconds for SCP iteration
  • res_tol - deviation tolerance past which solution is accepted (measure of convergence)
  • slew_rate - the quadratic penalty between time-consecutive controls (encourages smooth controls)
  • u_slew - the previous action taken to align the first plan action with (useful for smooth receding horizon control)

Additional Dynamics Settings

  • X_prev - previous state solution (guess), $x^{(i)} ~~ \forall i \in [1, \dots, N]$, shape = (N, xdim)
  • U_prev - previous control solution (guess), $u^{(i)} ~~ \forall i \in [0, \dots, N - 1]$, shape = (N, udim)
  • x_l - state lower box constraints, $x^{(i)} ~~ \forall i \in [1, \dots, N]$, shape = (N, xdim)
  • x_u - state upper box constraints, $x^{(i)} ~~ \forall i \in [1, \dots, N]$, shape = (N, xdim)
  • u_l - control lower box constraints, $u^{(i)} ~~ \forall i \in [0, \dots, N - 1]$, shape = (N, udim)
  • u_u - control upper box constraints, $u^{(i)} ~~ \forall i \in [0, \dots, N - 1]$, shape = (N, udim)

Nonlinear Cost and Constraints

The solver supports custom arbitrary cost via each-SCP-iteration cost linearization and custom constraints via each-SCP-iteration constraint reformulation into any convex-cone constraint.

  • lin_cost_fn is an optional callable which allows specifying a custom cost, it should take arbitrary X, U and return a tuple
    • cx, the linearization of the cost with respect to the state, np.shape(cx) == (N, xdim) or cx is None
    • cu, the linearization of the cost with respect to the controls, np.shape(cu) == (N, udim) or cu is None

I highly recommend using an auto-diff library to produce the linearizations to avoid unnecessary bugs.

  • extra_cstrs_fns is an optional callable which returns a list of conic constraints given arbitrary X, U
    • a conic constraint $G z - h \in \mathcal{K}$, (e.g., $A z - b \leq 0$) consists of a tuple of 8 elements
      • l - the number of non-positive orthant constraints (an integer)
      • q - a list with the sizes of second order cone constraints (list of integers)
      • e - the number of exponential cone constraints (integer)
      • G_left - the G matrix referring to existing variables
      • G_right - the G matrix for new variables to introduce
      • h - the right hand side vector for the constraint
      • c_left - the additive linear cost augmentation for existing variables
      • c_right - the linear (minimization) cost for new variables to introduce

Note: lin_cost_fn is expected to return a tuple, but extra_cstrs_fns must be a list of functions!

Variable Layout

The variable layout is control variables (# = N udim) followed by state variables (# = N xdim).

For consensus optimization the layout is

  • consensus controls (# = Nc udim)
  • free controls (# = M (N - Nc) udim)
  • states (# = M N xdim)

Misc Settings

  • solver_settings - a dictionary of settings to pass to the lower-level Julia solver
  • solver_state - a previous solver state to pass to the lower-level Julia solver
  • filter_method - whether to apply SCP solution low-pass-like filtering to avoid SCP solution oscillation
  • filter_window - how large a SCP iteration window to pick for SCP solution filtering
  • filter_it0 - what iteration of SCP to start applying filtering from
  • return_min_viol - whether to return the minimum deviation solution (measured from previous), not the last one
  • min_viol_it0 - what iteration to start looking for minimum deviation solution

Advanced Usage

Multiple solver processes

Launching multiple solver processes can be useful when many problems need to be solved. This presents some challenges

  • the solver needs to be able to discover existing, ready persistent worker processes
  • batch of problem specifications need to be fed to a solution function
    • we redefine the optimal control problem slightly as a set of keyword arguments for the solve function, positional arguments are now keywords arguments

Launch

$ python3 -m pmpc.remote --help

usage: remote.py [-h] [--port PORT] [--verbose] [--worker-num WORKER_NUM]
                 [--redis-host REDIS_HOST] [--redis-port REDIS_PORT]
                 [--redis-password REDIS_PASSWORD]

optional arguments:
  -h, --help            show this help message and exit
  --port PORT, -p PORT  TCP port on which to start the server
  --verbose, -v
  --worker-num WORKER_NUM, -n WORKER_NUM
                        Number of workers to start, 0 means number equal to
                        physical CPU cores.
  --redis-host REDIS_HOST
                        Redis host
  --redis-port REDIS_PORT
                        Redis port
  --redis-password REDIS_PASSWORD
                        Redis password

$ python3 -m pmpc.remote --worker-num 0 --redis-password my_redis_password

then

from pmpc.remote import solve_problems

problems = [generate_problem(...) for _ in range(32)]
solutions = solve_problems(problems)

We solve the discovery problem by making use of redis, an in-memory key-value store (a database). You need to have a working redis-server running (e.g., sudo apt install redis-server).

Note: Since our worker processes are TCP based, they can be run on any network connected computer (which has the copy of the python environment). As long as we can ensure that all computers can access a redis database where workers register.

We solve the problem specification by redefining the optimal control problem slightly. All arguments to the solve function are now keyword arguments a problem is defined as a dictionary Dict[str, Any].

The solve_problems method solves a list of dictionary problem definitions and returns a list of solutions.

Consensus Optimization for Control under Uncertainty

For consensus control, simply

  • batch the system instantiations into an extra (first) dimension
    • all problem data now obtains a size M first dimensions
  • specify Nc as a field in solver_settings, the length of the consensus horizon

TODO more detailed explanation

Non-convex Cost Example

N, xdim, udim = 10, 3, 2
X_ref = np.random.rand(N, xdim)

def lin_cost_fn(X, U):
    # cost is np.sum((X - X_ref) ** 2)
    cx = 2 * (X - X_ref)
    cu = None
    return cx, cu

Arbitrary Constraints

Arbitrary convex (cone) constraints can be introduced in a canonical form via a callaback which recomputes them at every SCP iteration. This allows to encode non-convex constraints via their SCP convexification.

The canonical form for a convex problem is given by

$$ \begin{aligned} & \text{minimize} && c^T z \ & \text{such that} && A_i z \leq_{\mathcal{K}} b_i ~~ \forall i \end{aligned} $$

where $A z \leq_{\mathcal{K}} b$ refers to a cone constraint. The three supported cone constraints are

Linear Constraints

Simply $A z \leq b$

Second-order Cone Constraints (SOCP)

Mathematically $||A_{2:n} z - b_{2:n}||_2 \leq a_1^T z - b_1$ or in code

np.linalg.norm(A[1:, :] @ z - b[1:]) <= A[0, :].T @ z - b[0]

Exponential Cone Constraints

The exponential cone is defined as a 3 output matrix expression (the image of the cone matrix is of dimension 3). We follow the convention from JuMP.jl

$$ K_\text{exp} = \lbrace (x, y, z) \in \mathbb{R}^3 ~~~~ : ~~~~ y e^{x / y} \leq z, ~~~~ y \geq 0 \rbrace $$

for

$$A v - b = (x, y, z)$$

so $A \in \mathbb{R}^{n \times 3}$ and $b \in \mathbb{R}^3$.

Solver selection

The underlying convex solver can be selected by passing a composite keyword argument

sol = solve(
  ...,
  solver_settings = dict(solver="ecos") # for example, or "cosmo", "osqp", "mosek"
)
  • ECOS - FREE - very fast and very general (used by default)
  • OSQP - FREE - very fast, but only supports linear constraints
  • COSMO - FREE - very general, but it tends to run very slowly
  • Mosek - NOT FREE - extremely fast, but requires a (not free) license
    • requires: a Mosek license

Particle (consensus/contingency) optimization

TODO

Warm-start support

Warm-start in SCP MPC can refer to either

  • warm-starting the SCP procedure through a good X_prev, U_prev - this is supported
  • warm-starting the underlying convex solver - not supported

Warm-starting of the SCP procedure by providing a good X_prev, U_prev guess is supported and very much encouraged for good SCP performance!

Warm-starting of the underlying convex solver is currently not supported, as it does not lead to a noticeable performance improvement on problems we tested the solver on.

Citing This Work

@inproceedings{dyro2021particle,
  author = {Dyro, Robert and Harrison, James and Sharma, Apoorva and Pavone, M.},
  title = {{P}article {M}{P}{C} for {U}ncertain and {L}earning-{B}ased {C}ontrol},
  booktitle = {{I}{E}{E}{E}/{R}{J}{S} {I}nternational {C}onference on {I}ntelligent {R}{O}bots and {S}ystems},
  year = {2021},
}

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

pmpc-0.7.1-py3-none-any.whl (45.8 kB view details)

Uploaded Python 3

pmpc-0.7.1-cp312-cp312-manylinux_2_35_aarch64.whl (223.3 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.35+ ARM64

pmpc-0.7.1-cp312-cp312-manylinux_2_28_x86_64.whl (247.9 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.28+ x86-64

pmpc-0.7.1-cp311-cp311-manylinux_2_35_aarch64.whl (223.3 MB view details)

Uploaded CPython 3.11manylinux: glibc 2.35+ ARM64

pmpc-0.7.1-cp311-cp311-manylinux_2_28_x86_64.whl (247.9 MB view details)

Uploaded CPython 3.11manylinux: glibc 2.28+ x86-64

pmpc-0.7.1-cp310-cp310-manylinux_2_35_aarch64.whl (223.3 MB view details)

Uploaded CPython 3.10manylinux: glibc 2.35+ ARM64

pmpc-0.7.1-cp310-cp310-manylinux_2_28_x86_64.whl (247.8 MB view details)

Uploaded CPython 3.10manylinux: glibc 2.28+ x86-64

pmpc-0.7.1-cp39-cp39-manylinux_2_35_aarch64.whl (223.3 MB view details)

Uploaded CPython 3.9manylinux: glibc 2.35+ ARM64

pmpc-0.7.1-cp39-cp39-manylinux_2_28_x86_64.whl (247.8 MB view details)

Uploaded CPython 3.9manylinux: glibc 2.28+ x86-64

pmpc-0.7.1-cp38-cp38-manylinux_2_35_aarch64.whl (223.3 MB view details)

Uploaded CPython 3.8manylinux: glibc 2.35+ ARM64

pmpc-0.7.1-cp38-cp38-manylinux_2_28_x86_64.whl (247.8 MB view details)

Uploaded CPython 3.8manylinux: glibc 2.28+ x86-64

pmpc-0.7.1-cp37-cp37m-manylinux_2_35_aarch64.whl (223.3 MB view details)

Uploaded CPython 3.7mmanylinux: glibc 2.35+ ARM64

pmpc-0.7.1-cp37-cp37m-manylinux_2_28_x86_64.whl (247.8 MB view details)

Uploaded CPython 3.7mmanylinux: glibc 2.28+ x86-64

File details

Details for the file pmpc-0.7.1-py3-none-any.whl.

File metadata

  • Download URL: pmpc-0.7.1-py3-none-any.whl
  • Upload date:
  • Size: 45.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.13

File hashes

Hashes for pmpc-0.7.1-py3-none-any.whl
Algorithm Hash digest
SHA256 5c9eaaedafb1a820be8ded6d358dff07199c4f4e3565d4aae914b9642bdd9554
MD5 0e336b061c4e9407fb78f032f154d7ee
BLAKE2b-256 ddf6c332e3ad11a3a73c71278156b2cd98c139bf1b08abf55e92aa27fa1d9e38

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp312-cp312-manylinux_2_35_aarch64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp312-cp312-manylinux_2_35_aarch64.whl
Algorithm Hash digest
SHA256 5123a0597fa558e43c141ccb6c2ad2ee88b91d287663380f2c014beb9523d937
MD5 3f9b56f9d587f5f1c1816d22d6ddc7cb
BLAKE2b-256 84b93b0852e0931927b8a4ba3b59add8ffb18e1f013afdbe05fefdd61fb795b2

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp312-cp312-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp312-cp312-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 557393936920b5251ff1f82e7337f29c90e334ff92b69a303680b6648e667886
MD5 a96dc30e0430760c535b3cc82abbe03a
BLAKE2b-256 329b79377741ba5410fb47347063dbca5db4f4e28fcb99b14c8b0686e2c65d4d

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp311-cp311-manylinux_2_35_aarch64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp311-cp311-manylinux_2_35_aarch64.whl
Algorithm Hash digest
SHA256 2c0d3ee33fef7d4e28bbaab390f358c057aaf583468a28a3cdd16cc6fe9d2600
MD5 31b53e5762fa662a9e1462744dcfc090
BLAKE2b-256 74b95f186e29dc97fc29e3814a7a87da00381b3b56e519298d7135535b3fc6d6

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp311-cp311-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp311-cp311-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 6f126e2c3ce11a84f292bb4df74a1072f50d3736b5d184bf0711db327f448047
MD5 b1f618e0cd066558177dc0f06db30a76
BLAKE2b-256 0d3953a111be84e16c36bade1a40ea9659e24f6182ed9fed88c616920535b917

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp310-cp310-manylinux_2_35_aarch64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp310-cp310-manylinux_2_35_aarch64.whl
Algorithm Hash digest
SHA256 eb8b03028ee1a1f7ef8fb22073be5983e0c60d1ed995dcec590d19122906ce24
MD5 c1128500addff26858c30daa78c44ea4
BLAKE2b-256 4304f3ae76f40cb683cdf949b460e3382de02cb2e4fab6ebbc42f3441a3d8b41

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp310-cp310-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp310-cp310-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 48bb83a2dc74818bc71e6f67cd841c5678df17a7061e165c83ecdd27a3d87b54
MD5 8b1ee74e3f616ef53464c9b4d2d4a32c
BLAKE2b-256 11d0358875c405c9755741c5be716dd25ebd6a0c050bd201de5b258dd8d6d74f

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp39-cp39-manylinux_2_35_aarch64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp39-cp39-manylinux_2_35_aarch64.whl
Algorithm Hash digest
SHA256 1d8d64666acf8afb664ca531d2a3875fb245968b5066f1b355e231e8ee0c71ee
MD5 b0472f0daae141a07e55b46a32895710
BLAKE2b-256 b97e2eb98e18552d334fe3b75fe29cdd2fd4d11b39a01a7e427c7f2d816cb784

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp39-cp39-manylinux_2_28_x86_64.whl.

File metadata

  • Download URL: pmpc-0.7.1-cp39-cp39-manylinux_2_28_x86_64.whl
  • Upload date:
  • Size: 247.8 MB
  • Tags: CPython 3.9, manylinux: glibc 2.28+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.13

File hashes

Hashes for pmpc-0.7.1-cp39-cp39-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 5d14b49ea9a68e63726f103e003b23a646fefcc95987e7b240c0abfb1bf1ae96
MD5 8636a05f98f3b88eae3ae68d2e9713c7
BLAKE2b-256 779f94c43200d1b415bf787ee6419e073b8979e9e94a231173466a1ff64d610a

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp38-cp38-manylinux_2_35_aarch64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp38-cp38-manylinux_2_35_aarch64.whl
Algorithm Hash digest
SHA256 2f2ba9b6ba08a49bf24b6517884379e77b6bb0e6d226e411c49d166b81f83946
MD5 16e9073f7e9f871d9f9fd007ff3bec5c
BLAKE2b-256 847892e8f8303a3e6f141ee9b0cad796e382f712e9aab9358ebecec560fd8ce3

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp38-cp38-manylinux_2_28_x86_64.whl.

File metadata

  • Download URL: pmpc-0.7.1-cp38-cp38-manylinux_2_28_x86_64.whl
  • Upload date:
  • Size: 247.8 MB
  • Tags: CPython 3.8, manylinux: glibc 2.28+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.13

File hashes

Hashes for pmpc-0.7.1-cp38-cp38-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 2fb1fe4e7029ee59ce6ad1e80f3d102822326943a0aae62567b7c8d73ca97bbc
MD5 c7720fa51572b896ba4cd55a24d666f4
BLAKE2b-256 c1895d9ec72cb445027ca400c4c00256d120cccabc8dc25570884b07d8c6a48c

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp37-cp37m-manylinux_2_35_aarch64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp37-cp37m-manylinux_2_35_aarch64.whl
Algorithm Hash digest
SHA256 8598341adebd5c25760237927d918e0126a31ddf9dc965915626273e75493601
MD5 124e89ea57cea8fba297e1b66b2fd3da
BLAKE2b-256 d611dc4cd0c405678d584646551ef423e8491fda70d6a874d33ebb9c724c048c

See more details on using hashes here.

File details

Details for the file pmpc-0.7.1-cp37-cp37m-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for pmpc-0.7.1-cp37-cp37m-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 0877e42f2d95a7c43f4907b585f7b3df9b192ed0ec4e828ef186a91e007abc9d
MD5 706fca5fd0290e299984e696108361a1
BLAKE2b-256 e881ff683fb2702a9825fa53d0d7f831ab9e12ca034b498737726cea85501724

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page