# 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: """ User‑tunable 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.)" )