dgx-spark-playbooks/nvidia/portfolio-optimization/assets/setup/src/cvar_parameters.py

113 lines
4.1 KiB
Python
Raw Normal View History

2026-01-02 22:21:53 +00:00
# SPDX-FileCopyrightText: Copyright (c) 2023-2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved. # noqa
# SPDX-License-Identifier: Apache-2.0
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from dataclasses import dataclass
from typing import Optional, Union
import numpy as np
@dataclass
class CvarParameters:
"""
Usertunable parameters and constraint limits for CVaR optimization.
Most parameters are scalars. Weight bounds ``w_min`` / ``w_max`` can be:
- numpy arrays (length n_assets) for per-asset bounds
- dict mapping asset names to bounds
- float for uniform bounds across all assets
- None for no bounds
Optional constraints (T_tar, cvar_limit, cardinality) default to None when
not specified.
"""
# Weight / cash bounds
w_min: Union[np.ndarray, dict, float] = 1.0 # Lower bound for each risky weight
w_max: Union[np.ndarray, dict, float] = 0.0 # Upper bound for each risky weight
c_min: float = 0 # Lower bound for cash allocation
c_max: float = 1 # Upper bound for cash allocation
# Risk model Parameters
risk_aversion: float = 1 # λ penalty applied to CVaR inside objective
confidence: float = 0.95 # α in CVaR_α (e.g. 0.95 -> 95 % CVaR)
# Soft / hard constraint targets
L_tar: float = 1.6 # Leverage constraint (Σ|wᵢ|)
T_tar: Optional[float] = None # Turnover constraint
cvar_limit: Optional[float] = None # Hard CVaR limit (None means "no hard limit")
cardinality: Optional[int] = None # number of assets to be selected
group_constraints: Optional[list[dict]] = None
# Group constraints:
# [{'group_name': group_name,
# 'tickers': tickers
# 'weight_bounds': {'w_min': w_min, 'w_max': w_max}}]
def update_w_min(self, new_w_min: Union[np.ndarray, dict, float]):
self.w_min = new_w_min
def update_w_max(self, new_w_max: Union[np.ndarray, dict, float]):
if new_w_max <= 1:
self.w_max = new_w_max
else:
raise ValueError("Invalid upper bound for weights!")
def update_c_min(self, new_c_min: float):
if new_c_min >= 0:
self.c_min = new_c_min
else:
raise ValueError("Cash should be non-negative!")
def update_c_max(self, new_c_max: float):
if new_c_max >= 0 and new_c_max <= 1:
self.c_max = new_c_max
else:
raise ValueError("Invalid upper bound for cash!")
def update_z_min(self, new_c_min: float):
self.z_min = new_c_min
def update_z_max(self, new_z_max: float):
self.z_max = new_z_max
def update_T_tar(self, new_T_tar: float):
self.T_tar = new_T_tar
def update_L_tar(self, new_L_tar: float):
self.L_tar = new_L_tar
def update_cvar_limit(self, new_cvar_limit: float):
self.cvar_limit = new_cvar_limit
def update_cardinality(self, new_cardinality: int):
if new_cardinality is None or (
isinstance(new_cardinality, int) and new_cardinality > 0
):
self.cardinality = new_cardinality
else:
raise ValueError("Cardinality must be a positive integer or None")
def update_risk_aversion(self, new_risk_aversion: float):
if new_risk_aversion >= 0:
self.risk_aversion = new_risk_aversion
else:
raise ValueError("Invalid risk aversion")
def update_confidence(self, new_confidence: float):
if new_confidence > 0 and new_confidence <= 1:
self.confidence = new_confidence
else:
raise ValueError(
"Invalid confidence level (should be between 0 and 1, "
"e.g. 95%, 99%, etc.)"
)