# Random Forest with Tree Selection Monte Carlo Based (RF-TSMC)
![](forest.png)
<a href="https://pypi.org/project/random-forest-mc"><img src="https://img.shields.io/pypi/pyversions/random-forest-mc" alt="Python versions"></a>
<a href="https://pypi.org/project/random-forest-mc"><img src="https://img.shields.io/pypi/v/random-forest-mc?color=blue" alt="PyPI version"></a>
![](https://img.shields.io/badge/Coverage-100%25-green)
![](https://img.shields.io/badge/Status-Stable-green)
[![](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/psf/black)
![](https://img.shields.io/badge/CodeQL-0%20alerts-green)
This project is about use Random Forest approach for *multiclass classification* using a dynamic tree selection Monte Carlo based.
## Description
This version of Random Forest makes the selection of trees based on indirect optimization via Monte Carlo simulations. Highly robust against unbalance between classes and missing data (for training and prediction). Optimized for training in parallel processing. Merge native between separate trained models, with simple merge or optimized; thus new models can be mixed with old models generating mixed models (mixing of decision trees), reducing possible new data vieses.
This model is highly recommended for EDAs, because it offers a high degree of explainability for PoCs, as it has an excellent relationship between generalization and cost of training and maintenance.
Already validated in production, with real data, for churn prediction, with data in the form of time series. In addition, it is excellent for studying Python, because the model is 100% Python code, using some of the most pytonic features! Inspect the code and enjoy!
The first implementation is found in [2] (using Common Lisp).
## Install:
Install using `pip`:
```bash
$ pip3 install random-forest-mc
```
Install from this repo (check version, `dev` version possible):
```bash
$ git clone https://github.com/ysraell/random-forest-mc.git
$ cd random-forest-mc
$ pip3 install .
```
## Usage:
-------
### Example of a full cycle using `titanic.csv`:
```python
import numpy as np
import pandas as pd
from random_forest_mc.model import RandomForestMC
from random_forest_mc.utils import (
LoadDicts, load_file_json, dump_file_json
)
dicts = LoadDicts("tests/")
dataset_dict = dicts.datasets_metadata
ds_name = "titanic"
params = dataset_dict[ds_name]
target_col = params["target_col"]
dataset = (
pd.read_csv(params["csv_path"])[params["ds_cols"] + [params["target_col"]]]
.dropna()
.reset_index(drop=True)
)
dataset["Age"] = dataset["Age"].astype(np.uint8)
dataset["SibSp"] = dataset["SibSp"].astype(np.uint8)
dataset["Pclass"] = dataset["Pclass"].astype(str)
dataset["Fare"] = dataset["Fare"].astype(np.uint32)
cls = RandomForestMC(
n_trees=8, target_col=target_col, max_discard_trees=4
)
cls.process_dataset(dataset)
cls.fit() # or with cls.fitParallel(max_workers=8)
y_test = dataset[params["target_col"]].to_list()
cls.setWeightedTrees(True) # predictions weighted by survive scores
y_pred = cls.testForest(dataset)
accuracy_hard = sum([v == p for v, p in zip(y_test, y_pred)]) / len(y_pred)
cls.setSoftVoting(True) # for predicitons using soft voting strategy
y_pred = cls.testForest(dataset)
accuracy_soft = sum([v == p for v, p in zip(y_test, y_pred)]) / len(y_pred)
# Simply predictions:
# One row
row = dataset.loc[0]
cls.predict(row)
{'0': 0.75, '1': 0.25}
# Multiple rows (dataset)
cls.predict(dataset.sample(n=10))
['0', '1', ...]
# Get the probabilities:
cls.predict_proba(dataset.sample(n=10))
[
{'0': 0.75, '1': 0.25},
{'0': 1.0, '1': 0.0},
...
{'0': 0.625, '1': 0.375}
]
```
-------
### Works with missing values:
```python
cols = list(dataset.columns)
cols.pop(cols.index('Class'))
ds = dataset[cols[:10]+['Class']]
row = ds.loc[0]
cls.predict(row)
{'0': 0.75, '1': 0.25}
cls.predict(ds.sample(n=10))
['0', '1', ...]
```
-------
### Save and load model:
```python
# Saving model:
ModelDict = cls.model2dict()
dump_file_json(path_dict, ModelDict)
del ModelDict
# Loading model
ModelDict = load_file_json(path_dict)
cls = RandomForestMC()
cls.dict2model(ModelDict)
# Before run fit again, load dataset. Check if the features are the same!
cls.process_dataset(dataset)
```
-------
### Feature counting and importance:
```python
# Feature counting (how much features in each tree):
cls.featCount()
(
(3.5, 0.5, 3, 4), # (mean, std, min, max)
[3, 4, 3, 4, 3, 4] # List of counting of features in each tree.
)
# Feature counting considering a given sample (using only the trees that predicted correctly):
row = dataset.loc[0]
cls.sampleClassFeatCount(row, row[target_col])
(
(3.5, 0.5, 3, 4), # (mean, std, min, max)
[3, 4, 3, 4, 3, 4] # List of counting of features in each tree.
)
# The follow methods have the same for a given sample in the
format `cls.sampleClass...(row, row[target_col])`.
# Feature importance:
cls.featImportance() # or cls.sampleClassFeatImportance(row, row[target_col])
{
'feat 1': 0.900000,
'feat 2': 0.804688,
'feat 3': 0.398438,
...
}
# Permutation feature importance:
cls.featPairImportance() # or cls.sampleClassFeatPairImportance(row, row[target_col])
{
('feat 1', 'feat 2'): 0.12,
('feat 1', 'feat 3'): 0.13,
('feat 2', 'feat 3'): 0.23,
...
}
# Permutation feature importance in matrix (dataframe):
cls.featCorrDataFrame() # or cls.sampleClassFeatCorrDataFrame(row, row[target_col])
feat 1 feat 2 feat 3
feat 1 0.900000 0.120000 0.130000
feat 2 0.120000 0.804688 0.230000
feat 3 0.130000 0.230000 0.398438
```
-------
### For merge different models (forests):
```python
...
cls.fit()
cls2.fit()
# Simply add all trees from cls2 in cls.
cls.mergeForest(cls2)
# Merge all trees from both models and keep the trees with scores within the top N survived scores.
cls.mergeForest(cls2, N, 'score')
# Merge all trees from both models and keep N random trees.
cls.mergeForest(cls2, N, 'random')
```
-------
### Predicting missing values:
Data with missing data (`nan` values):
`dataset_missing_values = `
| | Pclass | Sex | Age | SibSp | Fare | Embarked | Survived |
|---:|---------:|:------|------:|--------:|-------:|:-----------|-----------:|
| 0 | 1 | `nan` | 45 | 0 | 28 | S | 0 |
| 1 | 1 | male | 34 | 0 | 26 | `nan` | 1 |
Dictionary of features and possible values:
`dict_values = `
```json
{
"Sex": ["male", "female"],
"Embarked": ["S", "C", "Q"]
}
```
`cls.predictMissingValues(dataset_missing_values, dict_values)` returns:
| | Pclass | Sex | Age | SibSp | Fare | Embarked | Survived | 0 | 1 | row_id |
|---:|---------:|:-------|------:|--------:|-------:|:-----------|-----------:|---------:|---------:|---------:|
| 0 | 1 | `nan` | 45 | 0 | 28 | S | 0 | `nan` | `nan` | 0 |
| 1 | 1 | male | 45 | 0 | 28 | S | 0 | 0.5625 | 0.4375 | 0 |
| 2 | 1 | female | 45 | 0 | 28 | S | 0 | 0.0625 | 0.9375 | 0 |
| 3 | 1 | male | 34 | 0 | 26 | `nan` | 1 | `nan` | `nan` | 1 |
| 4 | 1 | male | 34 | 0 | 26 | S | 1 | 0.5625 | 0.4375 | 1 |
| 5 | 1 | male | 34 | 0 | 26 | C | 1 | 0.5625 | 0.4375 | 1 |
| 6 | 1 | male | 34 | 0 | 26 | Q | 1 | 0.5625 | 0.4375 | 1 |
You can see three new columns: in this case, you see one column (with the probabilities) for each target value (so if you have `n` different target values, you'll have `n` columns), and the last one named `row_id` linking all predicted ones to the first row (the original row with the missing value).
Observations:
- For the first row (`row_id = 0`) we have the highest propability (`0.9375`) when `sex = female`. We can predict this value as the best prediction for this missing value. Or, if we know the target (a truth value for the target), we can choose the value with the highest probability that match with the correct target value. The same is not possible to do with the second row (`row_id = 1`), doesn't matter which value predicted, the final probilities will be the same.
- Let the target column is optionally.
-------
### Notes:
- The value in the classes must be converted to `str` before make predicts.
- `fit` always add new trees (keep the trees generated before).
-------
### LoadDicts:
LoadDicts works loading all `JSON` files inside a given path, creating an object helper to use this files as dictionaries.
For example:
```python
from random_forest_mc.utils import LoadDicts
# JSONs: path/data.json, path/metdada.json
dicts = LoadDicts("path/")
# you have: dicts.data and dicts.metdada as dictionaries
# And a list of dictionaries loaded in:
dicts.List
> ["data", "metdada"]
```
## Fundamentals:
- Based on Random Forest method principles: ensemble of models (decision trees).
- In bootstrap process:
- the data sampled ensure the balance between classes, for training and validation;
- the list of features used are randomly sampled (with random number of features and order).
- For each tree:
- fallowing the sequence of a given list of features, the data is splited half/half based on meadian value;
- the splitting process ends when the samples have one only class;
- validation process based on dynamic threshold can discard the tree.
- For use the forest:
- all trees predictions are combined as a vote;
- it is possible to use soft or hard-voting.
- Positive side-effects:
- possible more generalization caused by the combination of overfitted trees, each tree is highly specialized in a smallest and different set of feature;
- robustness for unbalanced and missing data, in case of missing data, the feature could be skipped without degrade the optimization process;
- in prediction process, a missing value could be dealt with a tree replication considering the two possible paths;
- the survived trees have a potential information about feature importance.
- Robust for mssing values in categorical features during prediction process.
### References
[2] [Laboratory of Decision Tree and Random Forest (`github/ysraell/random-forest-lab`)](https://github.com/ysraell/random-forest-lab). GitHub repository.
[3] Credit Card Fraud Detection. Anonymized credit card transactions labeled as fraudulent or genuine. Kaggle. Access: <https://www.kaggle.com/mlg-ulb/creditcardfraud>.
### Development Framework (optional)
- [My data science Docker image](https://github.com/ysraell/my-ds).
With this image you can run all notebooks and scripts Python inside this repository.
### TO-DO list.
For TO-DO list see `TODO.md`.
Raw data
{
"_id": null,
"home_page": "https://github.com/ysraell/random-forest-mc",
"name": "random-forest-mc",
"maintainer": null,
"docs_url": null,
"requires_python": "<3.13,>=3.8.1",
"maintainer_email": null,
"keywords": "random forest, random",
"author": "Israel Oliveira",
"author_email": "prof.israel@gmail.com",
"download_url": "https://files.pythonhosted.org/packages/f3/d5/7fe8c720fec65e304eb915fdf004cb5408680b5eaff4f852474b941dc81b/random_forest_mc-1.2.0.tar.gz",
"platform": null,
"description": "# Random Forest with Tree Selection Monte Carlo Based (RF-TSMC)\n![](forest.png)\n\n<a href=\"https://pypi.org/project/random-forest-mc\"><img src=\"https://img.shields.io/pypi/pyversions/random-forest-mc\" alt=\"Python versions\"></a>\n<a href=\"https://pypi.org/project/random-forest-mc\"><img src=\"https://img.shields.io/pypi/v/random-forest-mc?color=blue\" alt=\"PyPI version\"></a>\n![](https://img.shields.io/badge/Coverage-100%25-green)\n![](https://img.shields.io/badge/Status-Stable-green)\n[![](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/psf/black)\n![](https://img.shields.io/badge/CodeQL-0%20alerts-green)\n\nThis project is about use Random Forest approach for *multiclass classification* using a dynamic tree selection Monte Carlo based.\n\n## Description\n\nThis version of Random Forest makes the selection of trees based on indirect optimization via Monte Carlo simulations. Highly robust against unbalance between classes and missing data (for training and prediction). Optimized for training in parallel processing. Merge native between separate trained models, with simple merge or optimized; thus new models can be mixed with old models generating mixed models (mixing of decision trees), reducing possible new data vieses.\nThis model is highly recommended for EDAs, because it offers a high degree of explainability for PoCs, as it has an excellent relationship between generalization and cost of training and maintenance.\nAlready validated in production, with real data, for churn prediction, with data in the form of time series. In addition, it is excellent for studying Python, because the model is 100% Python code, using some of the most pytonic features! Inspect the code and enjoy!\n\nThe first implementation is found in [2] (using Common Lisp).\n\n## Install:\n\nInstall using `pip`:\n\n```bash\n$ pip3 install random-forest-mc\n```\n\nInstall from this repo (check version, `dev` version possible):\n\n```bash\n$ git clone https://github.com/ysraell/random-forest-mc.git\n$ cd random-forest-mc\n$ pip3 install .\n```\n\n## Usage:\n\n-------\n### Example of a full cycle using `titanic.csv`:\n\n```python\nimport numpy as np\nimport pandas as pd\n\nfrom random_forest_mc.model import RandomForestMC\nfrom random_forest_mc.utils import (\n LoadDicts, load_file_json, dump_file_json\n)\n\ndicts = LoadDicts(\"tests/\")\ndataset_dict = dicts.datasets_metadata\nds_name = \"titanic\"\nparams = dataset_dict[ds_name]\ntarget_col = params[\"target_col\"]\ndataset = (\n pd.read_csv(params[\"csv_path\"])[params[\"ds_cols\"] + [params[\"target_col\"]]]\n .dropna()\n .reset_index(drop=True)\n)\ndataset[\"Age\"] = dataset[\"Age\"].astype(np.uint8)\ndataset[\"SibSp\"] = dataset[\"SibSp\"].astype(np.uint8)\ndataset[\"Pclass\"] = dataset[\"Pclass\"].astype(str)\ndataset[\"Fare\"] = dataset[\"Fare\"].astype(np.uint32)\ncls = RandomForestMC(\n n_trees=8, target_col=target_col, max_discard_trees=4\n)\ncls.process_dataset(dataset)\ncls.fit() # or with cls.fitParallel(max_workers=8)\ny_test = dataset[params[\"target_col\"]].to_list()\ncls.setWeightedTrees(True) # predictions weighted by survive scores\ny_pred = cls.testForest(dataset)\naccuracy_hard = sum([v == p for v, p in zip(y_test, y_pred)]) / len(y_pred)\ncls.setSoftVoting(True) # for predicitons using soft voting strategy\ny_pred = cls.testForest(dataset)\naccuracy_soft = sum([v == p for v, p in zip(y_test, y_pred)]) / len(y_pred)\n\n# Simply predictions:\n\n# One row\nrow = dataset.loc[0]\ncls.predict(row)\n{'0': 0.75, '1': 0.25}\n\n# Multiple rows (dataset)\ncls.predict(dataset.sample(n=10))\n['0', '1', ...]\n\n# Get the probabilities:\ncls.predict_proba(dataset.sample(n=10))\n[\n {'0': 0.75, '1': 0.25},\n {'0': 1.0, '1': 0.0},\n ...\n {'0': 0.625, '1': 0.375}\n]\n```\n-------\n### Works with missing values:\n\n```python\n\ncols = list(dataset.columns)\ncols.pop(cols.index('Class'))\nds = dataset[cols[:10]+['Class']]\n\nrow = ds.loc[0]\ncls.predict(row)\n{'0': 0.75, '1': 0.25}\n\ncls.predict(ds.sample(n=10))\n['0', '1', ...]\n```\n-------\n### Save and load model:\n\n```python\n# Saving model:\nModelDict = cls.model2dict()\ndump_file_json(path_dict, ModelDict)\ndel ModelDict\n\n# Loading model\nModelDict = load_file_json(path_dict)\ncls = RandomForestMC()\ncls.dict2model(ModelDict)\n# Before run fit again, load dataset. Check if the features are the same!\ncls.process_dataset(dataset)\n\n```\n-------\n### Feature counting and importance:\n\n```python\n\n# Feature counting (how much features in each tree):\ncls.featCount() \n(\n (3.5, 0.5, 3, 4), # (mean, std, min, max)\n [3, 4, 3, 4, 3, 4] # List of counting of features in each tree.\n)\n\n# Feature counting considering a given sample (using only the trees that predicted correctly):\nrow = dataset.loc[0]\ncls.sampleClassFeatCount(row, row[target_col])\n(\n (3.5, 0.5, 3, 4), # (mean, std, min, max)\n [3, 4, 3, 4, 3, 4] # List of counting of features in each tree.\n)\n\n\n# The follow methods have the same for a given sample in the\n format `cls.sampleClass...(row, row[target_col])`.\n\n# Feature importance:\ncls.featImportance() # or cls.sampleClassFeatImportance(row, row[target_col])\n{\n 'feat 1': 0.900000,\n 'feat 2': 0.804688,\n 'feat 3': 0.398438,\n ...\n}\n\n# Permutation feature importance:\ncls.featPairImportance() # or cls.sampleClassFeatPairImportance(row, row[target_col])\n{\n ('feat 1', 'feat 2'): 0.12,\n ('feat 1', 'feat 3'): 0.13,\n ('feat 2', 'feat 3'): 0.23,\n ...\n}\n\n# Permutation feature importance in matrix (dataframe):\ncls.featCorrDataFrame() # or cls.sampleClassFeatCorrDataFrame(row, row[target_col])\n feat 1 feat 2 feat 3\nfeat 1 0.900000 0.120000 0.130000\nfeat 2 0.120000 0.804688 0.230000\nfeat 3 0.130000 0.230000 0.398438\n```\n-------\n### For merge different models (forests):\n\n```python\n...\ncls.fit()\ncls2.fit()\n\n# Simply add all trees from cls2 in cls.\ncls.mergeForest(cls2)\n\n# Merge all trees from both models and keep the trees with scores within the top N survived scores.\ncls.mergeForest(cls2, N, 'score')\n\n# Merge all trees from both models and keep N random trees.\ncls.mergeForest(cls2, N, 'random')\n\n```\n-------\n### Predicting missing values:\n\nData with missing data (`nan` values): \n`dataset_missing_values = `\n| | Pclass | Sex | Age | SibSp | Fare | Embarked | Survived |\n|---:|---------:|:------|------:|--------:|-------:|:-----------|-----------:|\n| 0 | 1 | `nan` | 45 | 0 | 28 | S | 0 |\n| 1 | 1 | male | 34 | 0 | 26 | `nan` | 1 |\n\nDictionary of features and possible values:\n\n`dict_values = `\n```json\n\n{\n \"Sex\": [\"male\", \"female\"],\n \"Embarked\": [\"S\", \"C\", \"Q\"]\n}\n\n```\n\n`cls.predictMissingValues(dataset_missing_values, dict_values)` returns:\n\n| | Pclass | Sex | Age | SibSp | Fare | Embarked | Survived | 0 | 1 | row_id |\n|---:|---------:|:-------|------:|--------:|-------:|:-----------|-----------:|---------:|---------:|---------:|\n| 0 | 1 | `nan` | 45 | 0 | 28 | S | 0 | `nan` | `nan` | 0 |\n| 1 | 1 | male | 45 | 0 | 28 | S | 0 | 0.5625 | 0.4375 | 0 |\n| 2 | 1 | female | 45 | 0 | 28 | S | 0 | 0.0625 | 0.9375 | 0 |\n| 3 | 1 | male | 34 | 0 | 26 | `nan` | 1 | `nan` | `nan` | 1 |\n| 4 | 1 | male | 34 | 0 | 26 | S | 1 | 0.5625 | 0.4375 | 1 |\n| 5 | 1 | male | 34 | 0 | 26 | C | 1 | 0.5625 | 0.4375 | 1 |\n| 6 | 1 | male | 34 | 0 | 26 | Q | 1 | 0.5625 | 0.4375 | 1 |\n\n\nYou can see three new columns: in this case, you see one column (with the probabilities) for each target value (so if you have `n` different target values, you'll have `n` columns), and the last one named `row_id` linking all predicted ones to the first row (the original row with the missing value).\nObservations:\n- For the first row (`row_id = 0`) we have the highest propability (`0.9375`) when `sex = female`. We can predict this value as the best prediction for this missing value. Or, if we know the target (a truth value for the target), we can choose the value with the highest probability that match with the correct target value. The same is not possible to do with the second row (`row_id = 1`), doesn't matter which value predicted, the final probilities will be the same.\n- Let the target column is optionally.\n\n-------\n### Notes:\n\n- The value in the classes must be converted to `str` before make predicts.\n- `fit` always add new trees (keep the trees generated before).\n-------\n### LoadDicts:\n\nLoadDicts works loading all `JSON` files inside a given path, creating an object helper to use this files as dictionaries.\n\nFor example:\n```python\nfrom random_forest_mc.utils import LoadDicts\n# JSONs: path/data.json, path/metdada.json\ndicts = LoadDicts(\"path/\")\n# you have: dicts.data and dicts.metdada as dictionaries\n# And a list of dictionaries loaded in:\ndicts.List\n> [\"data\", \"metdada\"]\n```\n\n\n## Fundamentals:\n\n- Based on Random Forest method principles: ensemble of models (decision trees).\n\n- In bootstrap process:\n\n - the data sampled ensure the balance between classes, for training and validation;\n\n - the list of features used are randomly sampled (with random number of features and order).\n\n- For each tree:\n\n - fallowing the sequence of a given list of features, the data is splited half/half based on meadian value;\n\n - the splitting process ends when the samples have one only class;\n\n - validation process based on dynamic threshold can discard the tree.\n\n- For use the forest:\n\n - all trees predictions are combined as a vote;\n\n - it is possible to use soft or hard-voting.\n\n- Positive side-effects:\n\n - possible more generalization caused by the combination of overfitted trees, each tree is highly specialized in a smallest and different set of feature;\n\n - robustness for unbalanced and missing data, in case of missing data, the feature could be skipped without degrade the optimization process;\n\n - in prediction process, a missing value could be dealt with a tree replication considering the two possible paths;\n\n - the survived trees have a potential information about feature importance.\n\n - Robust for mssing values in categorical features during prediction process.\n\n### References\n\n[2] [Laboratory of Decision Tree and Random Forest (`github/ysraell/random-forest-lab`)](https://github.com/ysraell/random-forest-lab). GitHub repository.\n\n[3] Credit Card Fraud Detection. Anonymized credit card transactions labeled as fraudulent or genuine. Kaggle. Access: <https://www.kaggle.com/mlg-ulb/creditcardfraud>.\n\n### Development Framework (optional)\n\n- [My data science Docker image](https://github.com/ysraell/my-ds).\n\nWith this image you can run all notebooks and scripts Python inside this repository.\n\n### TO-DO list.\n\nFor TO-DO list see `TODO.md`.\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "This project is about use Random Forest approach using a dynamic tree selection Monte Carlo based.",
"version": "1.2.0",
"project_urls": {
"Homepage": "https://github.com/ysraell/random-forest-mc",
"Repository": "https://github.com/ysraell/random-forest-mc"
},
"split_keywords": [
"random forest",
" random"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "8910c3522c5bd31c55ec74288d5427f3f49dbe829872898bc0572d0f1e63852d",
"md5": "73770844a3e42b4922c47d5317313ad6",
"sha256": "d216363dbc0ac4bf552ea1d4576cd1a20b7e75b3c3a7299ec32aa93f69544c8f"
},
"downloads": -1,
"filename": "random_forest_mc-1.2.0-py3-none-any.whl",
"has_sig": false,
"md5_digest": "73770844a3e42b4922c47d5317313ad6",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": "<3.13,>=3.8.1",
"size": 15965,
"upload_time": "2024-11-17T13:28:29",
"upload_time_iso_8601": "2024-11-17T13:28:29.562337Z",
"url": "https://files.pythonhosted.org/packages/89/10/c3522c5bd31c55ec74288d5427f3f49dbe829872898bc0572d0f1e63852d/random_forest_mc-1.2.0-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "f3d57fe8c720fec65e304eb915fdf004cb5408680b5eaff4f852474b941dc81b",
"md5": "13bbaf11ec9fec9e44d3b42a8d69a11c",
"sha256": "5856ec3f8bcd063ad54d97f434ba36c815ba7f8cc3a9171ef93ef448c29dd3b1"
},
"downloads": -1,
"filename": "random_forest_mc-1.2.0.tar.gz",
"has_sig": false,
"md5_digest": "13bbaf11ec9fec9e44d3b42a8d69a11c",
"packagetype": "sdist",
"python_version": "source",
"requires_python": "<3.13,>=3.8.1",
"size": 18263,
"upload_time": "2024-11-17T13:28:31",
"upload_time_iso_8601": "2024-11-17T13:28:31.724439Z",
"url": "https://files.pythonhosted.org/packages/f3/d5/7fe8c720fec65e304eb915fdf004cb5408680b5eaff4f852474b941dc81b/random_forest_mc-1.2.0.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-11-17 13:28:31",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "ysraell",
"github_project": "random-forest-mc",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"lcname": "random-forest-mc"
}