QSS: Quadratic-Separable Solver
Project description
QSS: Quadratic-Separable Solver
QSS solves problems of the form
$$\begin{equation*} \begin{array}{ll} \text{minimize} & (1/2) x^T P x + q^T x + r + g(x) \\ \text{subject to} & Ax = b \end{array} \end{equation*}$$
where $x \in \bf{R}^n$ is the decision variable being optimized over. The objective is defined by a positive definite matrix $P \in \bf{S}^n_+$, a vector $q \in \bf{R}^n$, a scalar $r \in \bf{R}$, and a $g$ that is separable in the entries of $x$, i.e., $g$ can be written as $$g(x) = \sum_{i=1}^n g_i(x_i).$$ The constraints are defined by a matrix $A \in \bf{R}^{m \times n}$ and a vector $b \in \bf{R}^m$.
To use QSS, the user must specify $P$, $q$, $r$, $A$, $b$, as well as the $g_i$ from a built-in collection of separable functions.
Installation
pip install qss
Usage
After installing qss, import it with
import qss
This will expose the QSS class which is used to instantiate a solver object:
solver = qss.QSS(data)
Use the solve() method when ready to solve:
results = solver.solve(eps_abs=1e-5,
eps_rel=1e-5,
alpha=1.4,
rho=0.1,
max_iter=np.inf,
precond=True,
warm_start=False,
reg=True,
use_iter_refinement=False,
verbose=False,
)
Parameters
data: dictionary with the following keys:-
'P','q','r','A','b'specify the quadratic part of the objective and the linear constraint as in the problem formulation above.'P'and'A'should bescipy.sparseCSC matrices or QSSLinearOperators (see below),'q'and'b'should benumpyarrays, and'r'should be a scalar.'A'and'b'can be excluded fromdataor set toNoneif the linear equality constraints are not needed. -
'g'is a list of separable function definitions. Each separable function is declared as a dictionary with the following keys:'g': string that corresponds to a valid separable function name (see below for a list of supported functions).'args':'weight'(default 1),'scale'(default 1),'shift'(default 0) allow the'g'function to be applied in a weighted manner to a shifted and scaled input. Some functions take additional arguments, see below.'range': tuple specifying the start index and end index that the function should be applied to.
Note that the zero function will be applied to any indices that don't have another function specified for them.
-
eps_abs: scalar specifying absolute tolerance.eps_abs: scalar specifying relative tolerance.alpha: scalar specifying overstep size.rho: scalar specifying ADMM step size.max_iter: maximum number of ADMM iterations to perform.precond: boolean specifying whether to perform matrix equilibration.warm_start: boolean specifying whether to warm start upon a repeat call ofsolve().reg: boolean specifying whether to regularize KKT matrix. May fail on certain problem instances if set toFalse.use_iter_refinement: boolean, only matters ifregisTrue. Helps mitigate some of the accuracy loss due to regularization.verbose: boolean specifying whether to print verbose output.
Returns
A list containing the following:
objective: the objective value attained by the solution found byqss.solution: the solution vector.
Separable functions
The following convex separable functions are supported ( $\mathcal{I}$ is the $\{0, \infty\}$ indicator function):
| Function | Parameters | $g_i(x)$ |
|---|---|---|
zero |
$0$ | |
abs |
$|x|$ | |
is_pos |
$\mathcal I(x \geq 0)$ | |
is_neg |
$\mathcal I(x \leq 0)$ | |
is_bound |
lb: lower bound (default 0), ub: upper bound (default 1) |
$\mathcal I(l \leq x \leq u)$ |
is_zero |
$\mathcal I(x = 0)$ | |
pos |
$\max\{x, 0\}$ | |
neg |
$\max\{-x, 0\}$ | |
quantile |
tau: scalar in $(0, 1)$ |
$0.5 |x| + (\tau - 0.5) x$ |
huber |
M: positive scalar |
$x^2 \text{ if } |x| \leq M, 2M|x| - M^2 \text{ else}$ |
The following nonconvex separable functions are supported:
| Function | Parameters | $g_i(x)$ |
|---|---|---|
card |
$0$ for $x = 0$; $1$ for $x \neq 0$ | |
is_int |
$\mathcal I(x \text{ is an integer})$ | |
is_finite_set |
S: Python list of scalars |
$\mathcal I(x \in S)$ |
is_bool |
$\mathcal I(x \in \{0,1\})$ |
The t (weight), a (scale), b (shift) parameters are used to shift and scale the above as follows: t * g(ax - b).
Example
Applying the Huber function to a shifted version of the first 100 entries:
[{"g": "huber", "args": {"M": 2, "shift": -5}, "range": (0, 100)}]
Abstract linear operators
QSS comes with built-in support for abstract linear operators via the qss.linearoperator.LinearOperator class (hereafter referred to simply as LinearOperator).
The easiest way to build a LinearOperator is via its constructor. The argument to the constructor should be a list of lists representing a block matrix, in which each block is one of the following:
- SciPy sparse matrix or
scipy.sparse.linalg.LinearOperatororqss.linearoperator.LinearOperatororNone.
As an example, a constraint matrix A could be built as follows:
from qss.linearoperator import LinearOperator
A = LinearOperator([
[None, F, -I],
[I, I, None]
])
Where F is a scipy.sparse.linalg.LinearOperator that implements the Fourier transform and I is a SciPy sparse identity matrix.
There are several helper functions available to facilitate the creation of LinearOperators, all accessible through qss.linearoperator:
block_diag(D): Returns a block diagonalLinearOperatorfromD, a list of linear operators (SciPy sparse matrix,scipy.sparse.linalg.LinearOperator, orqss.linearoperator.LinearOperator).hstack(D): Horizontally concatenates list of linear operatorsDinto a singleLinearOperator.vstack(D): Vertically concatenates a list of linear operatorsDinto a singleLinearOperator.
Left and right matrix multiplication between a LinearOperator and a NumPy array is supported. Multiplication between LinearOperators is currently not supported. Matrix-vector multiplication between a LinearOperator F and a NumPy array v can be achieved with F.matvec(v) or F @ v. Multiplication with the adjoint of F can be achieved with F.rmatvec(v) or v @ F.
Note that solve times may be slower when LinearOperators are involved. If either P or A is a LinearOperator, the linear KKT system central to the QSS algorithm is solved indirectly, as opposed to directly via a factorization.
Example
Nonnegative least squares is a problem of the form
$$\begin{equation*} \begin{array}{ll} \text{minimize} & (1/2) \Vert Gx - h \Vert_2^2 \\ \text{subject to} & x \geq 0. \end{array} \end{equation*} $$
qss can be used to solve this problem as follows:
import numpy as np
import scipy as sp
import qss
p = 100
n = 500
G = sp.sparse.random(n, p, density=0.2, format="csc")
h = np.random.rand(n)
data = {}
data["P"] = G.T @ G
data["q"] = -h.T @ G
data["r"] = 0.5 * h.T @ h
data["g"] = [{"g": "is_pos", "range": (0, p)}] # Enforce x >= 0
solver = qss.QSS(data)
objective, x = solver.solve()
print(objective)
Development
To create a virtual environment, run
python3 -m venv env
Activate it with
source env/bin/activate
Clone the qss repository, cd into it, and install qss in development mode:
pip install -e ./ -r requirements.txt
Finally, test to make sure the installation worked:
pytest tests/
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file qss-0.2.5.tar.gz.
File metadata
- Download URL: qss-0.2.5.tar.gz
- Upload date:
- Size: 2.3 MB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
46726813f28442d57567096aa9b13e6c425eec5c45b87c48d2f8e186afea420a
|
|
| MD5 |
33003a22f73ab570109031bf8f6a113b
|
|
| BLAKE2b-256 |
a49b1e85a7b89fd83d39c6ef30a8b39f70b37f3dc8d795cfcd0ac9247cf9037d
|
Provenance
The following attestation bundles were made for qss-0.2.5.tar.gz:
Publisher:
build.yml on cvxgrp/qss
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
qss-0.2.5.tar.gz -
Subject digest:
46726813f28442d57567096aa9b13e6c425eec5c45b87c48d2f8e186afea420a - Sigstore transparency entry: 933262074
- Sigstore integration time:
-
Permalink:
cvxgrp/qss@724e5d250a5d4e04781471bacdb477e93ce1252c -
Branch / Tag:
refs/tags/v0.2.5 - Owner: https://github.com/cvxgrp
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
build.yml@724e5d250a5d4e04781471bacdb477e93ce1252c -
Trigger Event:
release
-
Statement type:
File details
Details for the file qss-0.2.5-py3-none-any.whl.
File metadata
- Download URL: qss-0.2.5-py3-none-any.whl
- Upload date:
- Size: 29.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a5d70282e9b9ed6f57b88e86c76f399c04916a890cc5dab8c327c99a95410be8
|
|
| MD5 |
b7efb0ef616642ecffbf1a64beace1ca
|
|
| BLAKE2b-256 |
97c7c9779ab8e74935061fce0dfbbb295f5da27889b2fe3fd4843d62ba8ce408
|
Provenance
The following attestation bundles were made for qss-0.2.5-py3-none-any.whl:
Publisher:
build.yml on cvxgrp/qss
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
qss-0.2.5-py3-none-any.whl -
Subject digest:
a5d70282e9b9ed6f57b88e86c76f399c04916a890cc5dab8c327c99a95410be8 - Sigstore transparency entry: 933262113
- Sigstore integration time:
-
Permalink:
cvxgrp/qss@724e5d250a5d4e04781471bacdb477e93ce1252c -
Branch / Tag:
refs/tags/v0.2.5 - Owner: https://github.com/cvxgrp
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
build.yml@724e5d250a5d4e04781471bacdb477e93ce1252c -
Trigger Event:
release
-
Statement type: