mirror of
https://github.com/NVIDIA/dgx-spark-playbooks.git
synced 2026-04-23 18:33:54 +00:00
113 lines
4.1 KiB
Python
113 lines
4.1 KiB
Python
|
|
# 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.)"
|
|||
|
|
)
|