# PyBondLab
PyBondLab is a Python module designed for portfolio sorting and other tools tailored for empirical asset pricing research, with a particular focus on corporate bonds. It is part of the [Open Source Bond Asset Pricing project](https://openbondassetpricing.com/).
## Overview
PyBondLab provides tools for computing and evaluating investment strategies. It features look-ahead bias free data cleaning procedures to ensure the integrity and reliability of empirical results.
## Installation
You can install the latest release using pip
```bash
pip install PyBondLab
```
Or you can install the master branch from GitHub by cloning the repo and running setup
```bash
git clone https://github.com/GiulioRossetti94/PyBondLab.git
cd PyBondLab
python setup.py install
```
## Usage & Examples
For a more complete overview, see notebooks and scripts in [example](https://github.com/GiulioRossetti94/PyBondLab/tree/main/examples)
### Portfolio sorting
This example demonstrates how to implement a long-short investment strategy using the `PyBondLab` module, based on quintile sorting of corporate bond on credit ratings.
At each month $t$, corporate bonds are sorted into five portfolios according to their credit rating (`RATING_NUM` column).
The strategy involves:
1. **Long Position**: buy the bonds in the quintile portfolio containing bonds with the lowest credit rating (i.e., long bonds with higher default risk)
2. **Short Position**: sell (short-sell) the bonds in the quintile portfolio containing bonds with the highest credit rating (i.e., short/sell bonds with low default risk)
```python
import PyBondLab as pbl
import pandas as pd
import numpy as np
import wrds
# read bond dataset
# use a pandas DataFrame with columns:
# "RATING_NUM": the credit rating of bonds
# "ISSUE_ID": the identifier for each bond
# "date": the date
# "RET_L5M": bond returns used to compute portfolio returns
# "VW": weights used to compute value-weighted performnace
# Read in data directly from WRDS
# Assumes you have a valid WRDS account and have set-up your cloud access #
# See:
# https://wrds-www.wharton.upenn.edu/pages/support/programming-wrds/programming-python/python-wrds-cloud/
wrds_username = '' # Input your WRDS username
db = wrds.Connection(wrds_username = wrds_username )
tbl1 = db.raw_sql("""SELECT DATE, ISSUE_ID,CUSIP, RATING_NUM, RET_L5M,AMOUNT_OUTSTANDING,
TMT, N_SP, PRICE_L5M
FROM wrdsapps.bondret
""")
# Required because the WRDS data comes with "duplicates" in the index
# does not affect data, but the "index" needs to be re-defined #
tbl1 = tbl1.reset_index()
tbl1['index'] = range(1,(len(tbl1)+1))
# Format the data
tbl1.columns = tbl1.columns.str.upper()
tbl1['date'] = pd.to_datetime(tbl1['DATE'])
tbl1['AMOUNT_OUTSTANDING'] = np.abs(tbl1['AMOUNT_OUTSTANDING'])
tbl1['PRICE_L5M'] = np.abs(tbl1['PRICE_L5M'])
tbl1 = tbl1.sort_values(['ISSUE_ID','DATE'])
# WRDS data "starts" officially on "2002-08-31"
tbl1 = tbl1[tbl1['date'] >= "2002-08-31"]
# Column used for value weigted returns
tbl1['VW'] = (tbl1['PRICE_L5M'] * tbl1['AMOUNT_OUTSTANDING'])/1000
holding_period = 1 # holding period returns
n_portf = 5 # number of portfolios
sort_var1 = 'RATING_NUM' # sorting characteristic/variable
# Initialize the single sort strategy
single_sort = pbl.SingleSort(holding_period, sort_var1, n_portf)
# Define a dictionary with strategy and optional parameters
params = {'strategy': single_sort,
'rating':None,
}
# Fit the strategy to the data. Specify ID identifier and column of returns
res = pbl.StrategyFormation(tbl1, **params).fit(IDvar = "ISSUE_ID",RETvar = "RET_L5M")
# Get long-short portfolios (equal- and value-weighted)
ew,vw = res.get_long_short()
```
### Data cleaning / filtering options
Currently, the package provides functionality for four different data cleaning procedures routinely applied in corporate bonds research.
Data cleaning functionalities are passed with a dictionary `{'adj':"{adjustments}","level":{level}, **options}`
#### Return exclusion (trimming)
Filtering out bonds whose returns are above/below a certain threshold level:
- `{'adj':'trim,'level':0.2}` excludes returns > 20%
- `{'adj':'trim,'level':-0.2}` excludes returns < 20%
- `{'adj':'trim,'level':[-0.2,0.2]}` excludes returns < -20% and returns >20%
#### Price exclusion
Filtering out bonds whose prices are above/below a certain threshold level. When constructing the strategy, either rename a column to `"PRICE"` or specify `PRICEvar` in the `.fit()` method.
- `{'adj':'price,'level':150}` excludes prices > 150
- `{'adj':'price,'level':20}` excludes prices < 150
- `{'adj':'price,'level':[20, 150]}` excludes prices < 20 and prices > 150
By default, the threshold for below/above exclusion is set to 25 (by default `'price_threshold':25`).
This implies that if `'level':26` is passed, bonds whose price is above 26 are excluded. To specify a different threshold, include `'price_threshold'` in the dictionary.
- `{'adj':'price,'level':26}` excludes prices >26
- `{'adj':'price,'level':26,'price_threshold':30}` excludes prices <26
#### Return bounce-back exclusion
Filtering out bonds where the product of their monthly returns $R_t \times R_{t-1}$ meets a pre-defined threshold.
- `{'adj':'bounce,'level':0.01}` excludes if $R_t \times R_{t-1}$ > 0.01
- `{'adj':'bounce,'level':-0.01}` excludes if $R_t \times R_{t-1}$ < 0.01
- `{'adj':'bounce,'level':[-0.01, 0.01]}` excludes if $R_t \times R_{t-1}$ <0.01 and $R_t \times R_{t-1}$ >0.01
#### Return winsorization
**Ex Ante Winsorization**: This affects the formation of long-short portfolios only for strategies that sort bonds into different bins based on signals derived from past returns (e.g., momentum, short-term reversal, long-term reversal). These returns are winsorized up until the portfolio formation month $t$, and the corresponding signal is then computed with these winsorized returns.
**Ex Post Winsorization**: This impacts the performance of any characteristic-sorted portfolio, as it modifies the bond returns used to compute the returns of the different portfolios. This introduces a look-ahead bias in the portfolio performance and makes the performance unattainable or 'infeasible'.
- `{'adj':'wins,'level':98,'location':'right'}`: winsorize the right tail of the distribution at the 98th percentile level.
- `{'adj':'wins,'level':98,'location':'left'}`: winsorize the left tail of the distribution at the 2nd percentile level.
- `{'adj':'wins,'level':98,'location':'both'}`: winsorize the left tail of the distribution at the 2nd percentile level and the right tail at 98th percentile level.
For ex ante winsorization, returns at time $t$ are winsorized based on the pooled distribution of returns from $t_0$ up until $t$. This means that for every $t$, percentile levels must be recomputed, which can impact the performance of the scripts. This is particularly relevant when running hundreds of different strategies (see Data Uncertainty section). To mitigate this issue, there is an option to pass a pandas DataFrame with precomputed rolling percentiles, thereby avoiding the need to compute the winsorization threshold at each iteration.
Example:
```python
BREAKPOINTS = pbl.load_breakpoints_WRDS()
BREAKPOINTS.index = pd.to_datetime(BREAKPOINTS.index)
# specify the parameters
params = {'strategy':rating_single_sort,'rating':'NIG',
'filters': {'adj':'wins','level': 98,'location':'both','df_breakpoints':BREAKPOINTS}}
```
### Data Uncertainty
The scripts [MomentumDataUncertainty.py](examples/MomentumDataUncertainty.py) and [RatingDataUncertainty.py](examples/MomentumDataUncertainty.py) provide replications of Section X in Dickerson, Robotti, and Rossetti, [2024](https://papers.ssrn.com/sol3/papers.cfm?abstract_id=4575879).
These scripts allow comparison of the effects of ex-ante and ex-post data cleaning procedures on the expected returns of long-short portfolios sorted by specific variables/characteristics. They highlight the look-ahead bias that is incorrectly introduced when ex-post cleaning procedures are applied.
### Portfolio statistics
## Portfolio Turnover
Portfolio turnover plays a crucial role in unveiling potential anomalies. Ignoring transaction costs and portfolio turnover might render a possible strategy more appealing by showing larger gross alphas and average expected returns. However, these strategies might not improve an investor's opportunity set once transaction costs and portfolio turnover are considered.
To illustrate this, let's consider the following Python code using the PyBondLab library. Before running this snippet, ensure you have executed the previous code snippet that sets up the necessary data and imports.
### Step-by-Step Instructions
1. **Execute Initial Setup**:
- Ensure you have run the initial setup code provided in an earlier part of this README, which includes importing libraries, defining the `single_sort` strategy, and loading the data into `tbl1`.
2. **Portfolio Turnover Calculation**:
- The following code snippet demonstrates how to compute portfolio turnover, accounting for the given strategy parameters (see [example](https://github.com/GiulioRossetti94/PyBondLab/tree/main/examples) for the complete script).
```python
# Define a dictionary with strategy and optional parameters.
params = {
'strategy': single_sort,
'rating': None,
'turnover': True
}
# Fit the strategy to the data. Specify ID identifier and column of returns.
res = pbl.StrategyFormation(tbl1, **params).fit(IDvar="ISSUE_ID", RETvar="RET_L5M")
# Get long-short portfolios (equal- and value-weighted).
ew, vw = res.get_long_short()
# Compute portfolio turnover.
ew_turnover, vw_turnover = res.get_ptf_turnover()
```
## References
Dickerson, Robotti, and Rossetti, [2024](https://papers.ssrn.com/sol3/papers.cfm?abstract_id=4575879)
Novy-Marx, Velikov, [2023](https://papers.ssrn.com/sol3/papers.cfm?abstract_id=4338007)
[Open Source Bond Asset Pricing project](https://openbondassetpricing.com/)
## Requirements
The following requirements were used for testing. It is possible that older versions would work as well
- Python(3.11+)
- NumPy(1.26+)
- Pandas(2.2+)
- statsmodels(0.14+)
- matplotlib(3+)
## Contact
Giulio Rossetti giulio.rossetti.1@wbs.ac.uk
Alex Dickerson alexander.dickerson1@unsw.edu.au
## License
This project is licensed under the MIT License. See the [LICENSE](LICENSE) file for details.
Raw data
{
"_id": null,
"home_page": "https://github.com/GiulioRossetti94/PyBondLab",
"name": "PyBondLab",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.11",
"maintainer_email": null,
"keywords": "corporate bonds, portfolio sorting, data cleaning",
"author": "Giulio Rossetti, Alex Dickerson",
"author_email": "Giulio.Rossetti.1@wbs.ac.uk, alexander.dickerson1@unsw.edu.au",
"download_url": "https://files.pythonhosted.org/packages/63/60/51c4364d516153fada22e88ca7a271939f327ee0c9896e8e5e4fced7c721/PyBondLab-0.1.4.tar.gz",
"platform": null,
"description": "# PyBondLab\nPyBondLab is a Python module designed for portfolio sorting and other tools tailored for empirical asset pricing research, with a particular focus on corporate bonds. It is part of the [Open Source Bond Asset Pricing project](https://openbondassetpricing.com/).\n\n## Overview\nPyBondLab provides tools for computing and evaluating investment strategies. It features look-ahead bias free data cleaning procedures to ensure the integrity and reliability of empirical results.\n\n## Installation\nYou can install the latest release using pip\n```bash\npip install PyBondLab\n```\n\nOr you can install the master branch from GitHub by cloning the repo and running setup\n\n```bash\ngit clone https://github.com/GiulioRossetti94/PyBondLab.git\ncd PyBondLab\npython setup.py install\n```\n\n## Usage & Examples\nFor a more complete overview, see notebooks and scripts in [example](https://github.com/GiulioRossetti94/PyBondLab/tree/main/examples)\n\n### Portfolio sorting\nThis example demonstrates how to implement a long-short investment strategy using the `PyBondLab` module, based on quintile sorting of corporate bond on credit ratings.\n\nAt each month $t$, corporate bonds are sorted into five portfolios according to their credit rating (`RATING_NUM` column). \nThe strategy involves:\n\n1. **Long Position**: buy the bonds in the quintile portfolio containing bonds with the lowest credit rating (i.e., long bonds with higher default risk)\n2. **Short Position**: sell (short-sell) the bonds in the quintile portfolio containing bonds with the highest credit rating (i.e., short/sell bonds with low default risk)\n\n```python\nimport PyBondLab as pbl\nimport pandas as pd\nimport numpy as np\nimport wrds\n\n# read bond dataset\n# use a pandas DataFrame with columns:\n# \"RATING_NUM\": the credit rating of bonds\n# \"ISSUE_ID\": the identifier for each bond\n# \"date\": the date\n# \"RET_L5M\": bond returns used to compute portfolio returns\n# \"VW\": weights used to compute value-weighted performnace\n\n# Read in data directly from WRDS\n# Assumes you have a valid WRDS account and have set-up your cloud access #\n# See:\n# https://wrds-www.wharton.upenn.edu/pages/support/programming-wrds/programming-python/python-wrds-cloud/\nwrds_username = '' # Input your WRDS username\ndb = wrds.Connection(wrds_username = wrds_username )\n\ntbl1 = db.raw_sql(\"\"\"SELECT DATE, ISSUE_ID,CUSIP, RATING_NUM, RET_L5M,AMOUNT_OUTSTANDING,\n TMT, N_SP, PRICE_L5M \n FROM wrdsapps.bondret\n \"\"\")\n \n# Required because the WRDS data comes with \"duplicates\" in the index\n# does not affect data, but the \"index\" needs to be re-defined # \ntbl1 = tbl1.reset_index()\ntbl1['index'] = range(1,(len(tbl1)+1))\n\n# Format the data\ntbl1.columns = tbl1.columns.str.upper()\ntbl1['date'] = pd.to_datetime(tbl1['DATE'])\ntbl1['AMOUNT_OUTSTANDING'] = np.abs(tbl1['AMOUNT_OUTSTANDING'])\ntbl1['PRICE_L5M'] = np.abs(tbl1['PRICE_L5M'])\ntbl1 = tbl1.sort_values(['ISSUE_ID','DATE'])\n\n# WRDS data \"starts\" officially on \"2002-08-31\"\ntbl1 = tbl1[tbl1['date'] >= \"2002-08-31\"]\n\n# Column used for value weigted returns\ntbl1['VW'] = (tbl1['PRICE_L5M'] * tbl1['AMOUNT_OUTSTANDING'])/1000\n\nholding_period = 1 # holding period returns\nn_portf = 5 # number of portfolios\nsort_var1 = 'RATING_NUM' # sorting characteristic/variable\n\n# Initialize the single sort strategy\nsingle_sort = pbl.SingleSort(holding_period, sort_var1, n_portf)\n\n# Define a dictionary with strategy and optional parameters\nparams = {'strategy': single_sort,\n 'rating':None,\n }\n\n# Fit the strategy to the data. Specify ID identifier and column of returns \nres = pbl.StrategyFormation(tbl1, **params).fit(IDvar = \"ISSUE_ID\",RETvar = \"RET_L5M\")\n\n# Get long-short portfolios (equal- and value-weighted)\new,vw = res.get_long_short()\n```\n### Data cleaning / filtering options\nCurrently, the package provides functionality for four different data cleaning procedures routinely applied in corporate bonds research.\nData cleaning functionalities are passed with a dictionary `{'adj':\"{adjustments}\",\"level\":{level}, **options}`\n\n#### Return exclusion (trimming)\nFiltering out bonds whose returns are above/below a certain threshold level:\n- `{'adj':'trim,'level':0.2}` excludes returns > 20%\n- `{'adj':'trim,'level':-0.2}` excludes returns < 20%\n- `{'adj':'trim,'level':[-0.2,0.2]}` excludes returns < -20% and returns >20%\n\n#### Price exclusion\nFiltering out bonds whose prices are above/below a certain threshold level. When constructing the strategy, either rename a column to `\"PRICE\"` or specify `PRICEvar` in the `.fit()` method.\n\n- `{'adj':'price,'level':150}` excludes prices > 150\n- `{'adj':'price,'level':20}` excludes prices < 150\n- `{'adj':'price,'level':[20, 150]}` excludes prices < 20 and prices > 150\n\nBy default, the threshold for below/above exclusion is set to 25 (by default `'price_threshold':25`).\nThis implies that if `'level':26` is passed, bonds whose price is above 26 are excluded. To specify a different threshold, include `'price_threshold'` in the dictionary.\n\n- `{'adj':'price,'level':26}` excludes prices >26\n- `{'adj':'price,'level':26,'price_threshold':30}` excludes prices <26\n\n#### Return bounce-back exclusion\nFiltering out bonds where the product of their monthly returns $R_t \\times R_{t-1}$ meets a pre-defined threshold.\n\n- `{'adj':'bounce,'level':0.01}` excludes if $R_t \\times R_{t-1}$ > 0.01\n- `{'adj':'bounce,'level':-0.01}` excludes if $R_t \\times R_{t-1}$ < 0.01 \n- `{'adj':'bounce,'level':[-0.01, 0.01]}` excludes if $R_t \\times R_{t-1}$ <0.01 and $R_t \\times R_{t-1}$ >0.01\n\n#### Return winsorization\n**Ex Ante Winsorization**: This affects the formation of long-short portfolios only for strategies that sort bonds into different bins based on signals derived from past returns (e.g., momentum, short-term reversal, long-term reversal). These returns are winsorized up until the portfolio formation month $t$, and the corresponding signal is then computed with these winsorized returns.\n\n**Ex Post Winsorization**: This impacts the performance of any characteristic-sorted portfolio, as it modifies the bond returns used to compute the returns of the different portfolios. This introduces a look-ahead bias in the portfolio performance and makes the performance unattainable or 'infeasible'. \n\n- `{'adj':'wins,'level':98,'location':'right'}`: winsorize the right tail of the distribution at the 98th percentile level.\n- `{'adj':'wins,'level':98,'location':'left'}`: winsorize the left tail of the distribution at the 2nd percentile level.\n- `{'adj':'wins,'level':98,'location':'both'}`: winsorize the left tail of the distribution at the 2nd percentile level and the right tail at 98th percentile level.\n\nFor ex ante winsorization, returns at time $t$ are winsorized based on the pooled distribution of returns from $t_0$ up until $t$. This means that for every $t$, percentile levels must be recomputed, which can impact the performance of the scripts. This is particularly relevant when running hundreds of different strategies (see Data Uncertainty section). To mitigate this issue, there is an option to pass a pandas DataFrame with precomputed rolling percentiles, thereby avoiding the need to compute the winsorization threshold at each iteration.\nExample:\n```python\nBREAKPOINTS = pbl.load_breakpoints_WRDS()\nBREAKPOINTS.index = pd.to_datetime(BREAKPOINTS.index)\n\n# specify the parameters \nparams = {'strategy':rating_single_sort,'rating':'NIG',\n'filters': {'adj':'wins','level': 98,'location':'both','df_breakpoints':BREAKPOINTS}}\n\n```\n### Data Uncertainty\nThe scripts [MomentumDataUncertainty.py](examples/MomentumDataUncertainty.py) and [RatingDataUncertainty.py](examples/MomentumDataUncertainty.py) provide replications of Section X in Dickerson, Robotti, and Rossetti, [2024](https://papers.ssrn.com/sol3/papers.cfm?abstract_id=4575879).\n\nThese scripts allow comparison of the effects of ex-ante and ex-post data cleaning procedures on the expected returns of long-short portfolios sorted by specific variables/characteristics. They highlight the look-ahead bias that is incorrectly introduced when ex-post cleaning procedures are applied.\n\n### Portfolio statistics\n## Portfolio Turnover\n\nPortfolio turnover plays a crucial role in unveiling potential anomalies. Ignoring transaction costs and portfolio turnover might render a possible strategy more appealing by showing larger gross alphas and average expected returns. However, these strategies might not improve an investor's opportunity set once transaction costs and portfolio turnover are considered.\n\nTo illustrate this, let's consider the following Python code using the PyBondLab library. Before running this snippet, ensure you have executed the previous code snippet that sets up the necessary data and imports.\n\n### Step-by-Step Instructions\n\n1. **Execute Initial Setup**:\n - Ensure you have run the initial setup code provided in an earlier part of this README, which includes importing libraries, defining the `single_sort` strategy, and loading the data into `tbl1`.\n\n2. **Portfolio Turnover Calculation**:\n - The following code snippet demonstrates how to compute portfolio turnover, accounting for the given strategy parameters (see [example](https://github.com/GiulioRossetti94/PyBondLab/tree/main/examples) for the complete script). \n\n```python\n# Define a dictionary with strategy and optional parameters.\nparams = {\n 'strategy': single_sort,\n 'rating': None,\n 'turnover': True\n}\n\n# Fit the strategy to the data. Specify ID identifier and column of returns.\nres = pbl.StrategyFormation(tbl1, **params).fit(IDvar=\"ISSUE_ID\", RETvar=\"RET_L5M\")\n\n# Get long-short portfolios (equal- and value-weighted).\new, vw = res.get_long_short()\n\n# Compute portfolio turnover.\new_turnover, vw_turnover = res.get_ptf_turnover()\n```\n\n\n\n\n## References\nDickerson, Robotti, and Rossetti, [2024](https://papers.ssrn.com/sol3/papers.cfm?abstract_id=4575879)\n\nNovy-Marx, Velikov, [2023](https://papers.ssrn.com/sol3/papers.cfm?abstract_id=4338007)\n\n[Open Source Bond Asset Pricing project](https://openbondassetpricing.com/)\n\n## Requirements\nThe following requirements were used for testing. It is possible that older versions would work as well\n- Python(3.11+)\n- NumPy(1.26+)\n- Pandas(2.2+)\n- statsmodels(0.14+)\n- matplotlib(3+)\n\n## Contact\nGiulio Rossetti giulio.rossetti.1@wbs.ac.uk\n\nAlex Dickerson alexander.dickerson1@unsw.edu.au\n\n## License\nThis project is licensed under the MIT License. See the [LICENSE](LICENSE) file for details.\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "Performs portfolio sorting and strategy evaluation for corporate bonds",
"version": "0.1.4",
"project_urls": {
"Bug Tracker": "https://github.com/GiulioRossetti94/PyBondLab/issues",
"Homepage": "https://github.com/GiulioRossetti94/PyBondLab",
"Open Source Bond Asset Pricing": "https://openbondassetpricing.com/",
"Source Code": "https://github.com/GiulioRossetti94/PyBondLab"
},
"split_keywords": [
"corporate bonds",
" portfolio sorting",
" data cleaning"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "9a7834921c7f5da1ff90c7c175a210901941fc938c2c4bc3a43218cb9d4530ae",
"md5": "2e9df103b55d33bebd2dbb7e47447ba4",
"sha256": "2a8a3b21a125fd313c72b21120f597afb36a701c7f21dff3191fe860ed063477"
},
"downloads": -1,
"filename": "PyBondLab-0.1.4-py3-none-any.whl",
"has_sig": false,
"md5_digest": "2e9df103b55d33bebd2dbb7e47447ba4",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.11",
"size": 132338,
"upload_time": "2024-12-10T13:09:07",
"upload_time_iso_8601": "2024-12-10T13:09:07.766223Z",
"url": "https://files.pythonhosted.org/packages/9a/78/34921c7f5da1ff90c7c175a210901941fc938c2c4bc3a43218cb9d4530ae/PyBondLab-0.1.4-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "636051c4364d516153fada22e88ca7a271939f327ee0c9896e8e5e4fced7c721",
"md5": "016ee92823b7cec75c1f82078ebd642a",
"sha256": "fb03fc50fb1e8301ea99bbed916040b83dfe165fccc1207bfdfd5fe351b3dfb6"
},
"downloads": -1,
"filename": "PyBondLab-0.1.4.tar.gz",
"has_sig": false,
"md5_digest": "016ee92823b7cec75c1f82078ebd642a",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.11",
"size": 135421,
"upload_time": "2024-12-10T13:09:09",
"upload_time_iso_8601": "2024-12-10T13:09:09.755314Z",
"url": "https://files.pythonhosted.org/packages/63/60/51c4364d516153fada22e88ca7a271939f327ee0c9896e8e5e4fced7c721/PyBondLab-0.1.4.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-12-10 13:09:09",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "GiulioRossetti94",
"github_project": "PyBondLab",
"travis_ci": false,
"coveralls": false,
"github_actions": false,
"requirements": [
{
"name": "numpy",
"specs": [
[
">=",
"1.26"
]
]
},
{
"name": "pandas",
"specs": [
[
">=",
"2.2"
]
]
},
{
"name": "statsmodels",
"specs": [
[
">=",
"0.14"
]
]
},
{
"name": "matplotlib",
"specs": [
[
">=",
"3"
]
]
}
],
"lcname": "pybondlab"
}