# Mirabolic
Tools for statistical modeling and analysis, written by [Mirabolic](https://www.mirabolic.net/). These modules can be installed by running
```
pip install --upgrade mirabolic
```
and the source code can be found at https://github.com/Mirabolic/mirabolic
# Table of Contents
- [Visualizing LLM Embeddings](#visualizing-llm-embeddings)
- [Installation](#installation)
- [Examples](#examples)
- [Comparing Event Rates](#comparing-event-rates)
- [CDFs With Confidence Intervals](#cdfs-with-confidence-intervals)
- [Neural Nets for GLM regression](#neural-nets-for-glm-regression)
## Visualizing LLM Embeddings
Suppose we have a collection of paragraphs describing different things. We can use an LLM to "embed" those paragraphs, i.e., convert them into vectors in some high dimensional space. (In concrete terms: each paragraph is converted into a list of about 3000 numbers). Because of the magic of LLMs, similar paragraphs are supposed to cluster near each other in this high-dimensional space.
But how do we know if that's actually happening? Well, we might be lucky enough to have labels describing a few types paragraphs. If that's the case, we can map the embeddings down to two dimensions, color the points by label, and look at the scatter plot. If our embeddings are working correctly, points with the same color will group together.
### Installation
Because this code invokes LLMs, we require a little extra setup. In particular, we use OpenAI's embedding models, which means that you, dear reader, need an OpenAI API key. For those unfamiliar with this process, we offer a primer.
To get an OpenAI API key
1 [Get a developer account on OpenAI.](https://platform.openai.com/docs/overview) (You'll need a credit card, I'm afraid. FWIW, small experiments typically cost several cents.)
2 [Get an API key.](https://platform.openai.com/api-keys) This is a string of letters and numbers that begins `sk-`.
Now that you have the API key, you need to store it where your system can find it. This can be writing a file called `.env` and sticking in the API key. (If you already have a file like that, just add this as an extra line). The contents of your file should look like this:
```
OPENAI_API_KEY=sk-...
```
When you run this code, it will look for a `.env` file in the current directory; if it doesn't find it, it will look in the parent directory, and the parent's parent directory, and so on. This means that if you'd like, you can put the `.env` file in your home directory and code you execute will find it. If you're not running in a home directory, just make sure that there is a `.env` file in whatever directory you use.
A word of warning: this key should stay secret. If you're working in a Git repo, be sure to .gitignore the .env file so you do not accidentally commit it.
### Examples
We provide an example, but first, let's make some data. (Note that here our "paragraph" is only one or two words.)
```python
# Make some data that falls into natural classes
dog_breeds = ["Labrador Retriever", "German Shepherd", "Golden Retriever", "Bulldog", "Poodle", "Beagle", "Rottweiler", "Yorkshire Terrier", "Boxer", "Dachshund"]
cat_breeds = ["Siamese", "Persian", "Maine Coon", "Ragdoll", "Bengal", "Sphynx", "British Shorthair", "Abyssinian", "Scottish Fold", "Russian Blue"]
common_birds = ["Northern Cardinal", "American Robin", "Blue Jay", "House Sparrow", "Mourning Dove", "European Starling", "Black-capped Chickadee", "American Goldfinch", "Red-winged Blackbird", "Downy Woodpecker"]
data = dog_breeds + cat_breeds + common_birds
labels = len(dog_breeds) * ["dog"] + len(cat_breeds) * ["cat"] + len(common_birds) * ["birds"]
```
Next, let's use an LLM to embed the data in a high dimension, then visualize it in two dimensions:
```python
import mirabolic
embedder = mirabolic.llm_embedder(X=data, L=labels)
embedder.run_all()
```
![LLM_embedding_3](https://github.com/user-attachments/assets/d17b5967-a599-4969-81e6-12cef2ba7106)
Wow! Nice grouping of categories.
Sometimes we might want to process the text before embedding it. The `llm_embedder()` allows for arbitrary text transformations. Here's an example in which the transformation itself invokes another LLM.
```python
import mirabolic
def f(s):
"""
Replace the breed or species with a pet name.
"""
prompt = (
f"I have a pet {s}. I need a name for my pet that is "
"some sort of outrageous pun that will reveal its breed or species. "
"Please reply with one name (and no other text)."
)
response = mirabolic.LLM_API(prompt)
print(f"{s} => {response}")
return response
embedder = mirabolic.llm_embedder(X=data, L=labels, f=f)
embedder.run_all()
```
So, we transform the species name into a cutesy pet name (e.g., `Boxer` -> `Rocky Bal-bow-wow`, or `Russian Blue` -> `Czar Purrtin`). Of course, this makes it more difficult to distinguish the underlying species, so the clusters become murkier. Nevertheless, the cat/dog/bird regions are still fairly visible:
![LLM_embedding_2](https://github.com/user-attachments/assets/38c04782-5a58-4ed7-bf3c-8a1ec8d43476)
## Comparing Event Rates
Suppose you run a website. Every day you run a new email campaign to drive traffic to your site. You're considering a new approach, so you A/B test your campaigns: a portion of your emails use the new style of campaign and the rest use the old style.
Which style won? The new campaign looks good, but how do you know it's not just a random fluke? You vaguely remember an old stats teacher saying something about designing experiments with sufficient statistical power for a target effect size, but you don't know what the effect size is going to be, and you've already run the experiment! You really just want some simple way of visualizing the data you already have to see if the new style of campaigns are an improvement, or if it's all just random noise.
We provide such a tool here:
```python
import numpy as np
import mirabolic
import matplotlib.pyplot as plt
# Make some synthetic data
num_campaigns = 8
num_emails = 3000 # Number of emails in one arm of one campaign
# In this example, the "B" arm has a slightly higher conversion rate
num_conversions_A = np.random.binomial(num_emails, 0.03, size=num_campaigns)
num_conversions_B = np.random.binomial(num_emails, 0.04, size=num_campaigns)
num_emails_A = num_campaigns * [num_emails]
num_emails_B = num_campaigns * [num_emails]
# Plot the figure
plt.figure(figsize=(6, 6))
results = mirabolic.rate_comparison(
num_successes_A=num_conversions_A,
num_successes_B=num_conversions_B,
num_trials_A=num_emails_A,
num_trials_B=num_emails_B,
)
plt.show()
```
Here's the output:
![response_rate_1](https://github.com/user-attachments/assets/b43cdaca-7351-4e0a-b29f-73ced4d5a44c)
The figure shows a scatter plot with 8 points. Each point corresponds to a campaign, where the x-value is the conversion rate for the A arm (the old style, say) and the y-value is the conversion rate for the B arm (the new style). Around each point is a confidence rectangle showing how seriously to take it. If all the rectangles overlap with the diagonal line, then you don't have enough data to draw any conclusions (at least from individual campaigns). If the rectangles mostly fall above the dotted line, then the new style is an improvement; if below, it's making things worse.
## CDFs with Confidence Intervals
When exploring data, it can be very helpful to plot observations as a cumulative density function, or [CDF](https://en.wikipedia.org/wiki/Cumulative_distribution_function). (In medical contexts, doctors study the "survival function", which is 1 minus the CDF and essentially amounts to the same thing.) We plot a CDF by sorting the observed data from smallest to largest value. We can treat[^iid] the value in the middle of the sorted list as approximately the median, the value 90% of the way up the list as near the 90th percentile, and so forth.
[^iid]: We assume the data consists of i.i.d. draws from some unknown probability distribution.
When interpreting a CDF, or comparing two of them, one often wishes for something akin to a confidence interval. How close is the middle value to the median? Somewhat surprisingly, it is possible to compute the corresponding confidence intervals exactly.[^Beta]
[^Beta]: More precisely, suppose we draw a sample of n i.i.d. observations and consider the i-th smallest; if we are sampling from *any* continuous probability distribution, then the distribution of the corresponding quantile has a [Beta distribution](https://en.wikipedia.org/wiki/Beta_distribution), B(i, n-i+1).
For a single data point, the uncertainty around its quantile can be thought of as a confidence interval. If we consider all the data points, then we refer to a *confidence band*.[^Credible]
[^Credible]: Because we have access to a prior distribution on quantiles, these are arguably *[credible intervals](https://en.wikipedia.org/wiki/Credible_interval)* and *credible bands*, rather than confidence intervals and bands. We do not concern ourselves with this detail.
We provide a simple function for plotting CDFs with confidence bands; one invokes it by calling something like:
```
import mirabolic
import matplotlib.pyplot as plt
import random
data = [random.random() for _ in range(50)]
mirabolic.cdf_plot(data=data)
plt.show()
```
The figure looks like this:
![CDF_1](https://github.com/user-attachments/assets/dbfb9640-78c4-4593-baf2-1ee1de591285)
More examples can be found in [`mirabolic/cdf/sample_usage.py`](https://github.com/Mirabolic/mirabolic/blob/main/mirabolic/cdf/sample_usage.py).
## Neural Nets for GLM regression
GLMs ([Generalized Linear Models](https://en.wikipedia.org/wiki/Generalized_linear_model)) are a relatively broad class of statistical model first popularlized in the 1970s. These have grown popular in the actuarial literature as a method of predicting insurance claims costs and frequency.
With the appropriate loss function, GLMs can be expressed as neural nets. These two techniques have traditionally been treated as distinct, but bridging the divide provides two advantages.
First, a vast amount of effort has been spent on optimizing and accelerating neural nets over the past several years (GPUs and TPUs, parallelization). By expressing a GLM as a neural net, we can leverage this work.[^NN]
[^NN]: In terms of focus, [this chart](https://trends.google.com/trends/explore?geo=US&q=deep%20learning,actuarial%20science) suggests something of the explosion of interest in neural nets and deep learning relative to more traditional actuarial models.
Second, expressing a GLM as a neural net opens the possibility of extending the neural net before or after the GLM component. For instance, suppose we build three subnets that each computed a single feature, and then feed the three outputs as inputs into the Poisson regression net. This single larger network would allow the three subnets to engineer their individual features such that the loss function of the joint network was optimized. This approach provides a straightforward way of performing non-linear feature engineering but retaining the explainability of a GLM. This two-step approach may provide regulatory advantages, since US Departments of Insurance (DOIs) have been reluctant to approve end-to-end deep learning models.
We provide loss functions for several of the most commonly used GLMs. Minimal code might look something like this:
```
import mirabolic.neural_glm as neural_glm
from keras.models import Sequential
import tf
model = Sequential()
# Actually design your neural net...
# model.add(...)
loss=neural_glm.Poisson_link_with_exposure
optimizer = tf.keras.optimizers.Adam()
model.compile(loss=neural_glm, optimizer=optimizer)
```
To illustrate this process in more detail, we provide code to perform [Poisson regression](https://en.wikipedia.org/wiki/Poisson_regression) and Negative Binomial regression using a neural net.
To see the code in action, grab [the source code](https://github.com/Mirabolic/mirabolic) from GitHub, then [change to this directory](https://github.com/Mirabolic/mirabolic/tree/main/mirabolic/neural_glm), and run
```
python run_examples.py
```
This will generate Poisson-distributed data and corresponding features and then try to recover the "betas" (i.e., the linear coefficients of the GLM) using various models, outputting both the true and recovered values.
Raw data
{
"_id": null,
"home_page": "https://github.com/Mirabolic/mirabolic",
"name": "mirabolic",
"maintainer": null,
"docs_url": null,
"requires_python": null,
"maintainer_email": null,
"keywords": "Statistics, Machine Learning, CDF, Quantiles",
"author": "Bill Bradley",
"author_email": null,
"download_url": "https://files.pythonhosted.org/packages/87/a4/1cd4895bfc5d8f9e9644b7c1d017f2d56f213f9fe794317787da022b2e0b/mirabolic-0.3.10.tar.gz",
"platform": null,
"description": "# Mirabolic\nTools for statistical modeling and analysis, written by [Mirabolic](https://www.mirabolic.net/). These modules can be installed by running\n```\npip install --upgrade mirabolic\n```\nand the source code can be found at https://github.com/Mirabolic/mirabolic\n\n# Table of Contents\n- [Visualizing LLM Embeddings](#visualizing-llm-embeddings)\n - [Installation](#installation)\n - [Examples](#examples)\n- [Comparing Event Rates](#comparing-event-rates)\n- [CDFs With Confidence Intervals](#cdfs-with-confidence-intervals)\n- [Neural Nets for GLM regression](#neural-nets-for-glm-regression)\n\n## Visualizing LLM Embeddings\nSuppose we have a collection of paragraphs describing different things. We can use an LLM to \"embed\" those paragraphs, i.e., convert them into vectors in some high dimensional space. (In concrete terms: each paragraph is converted into a list of about 3000 numbers). Because of the magic of LLMs, similar paragraphs are supposed to cluster near each other in this high-dimensional space.\n\nBut how do we know if that's actually happening? Well, we might be lucky enough to have labels describing a few types paragraphs. If that's the case, we can map the embeddings down to two dimensions, color the points by label, and look at the scatter plot. If our embeddings are working correctly, points with the same color will group together. \n\n### Installation\nBecause this code invokes LLMs, we require a little extra setup. In particular, we use OpenAI's embedding models, which means that you, dear reader, need an OpenAI API key. For those unfamiliar with this process, we offer a primer.\n\nTo get an OpenAI API key\n1 [Get a developer account on OpenAI.](https://platform.openai.com/docs/overview) (You'll need a credit card, I'm afraid. FWIW, small experiments typically cost several cents.)\n2 [Get an API key.](https://platform.openai.com/api-keys) This is a string of letters and numbers that begins `sk-`.\n\nNow that you have the API key, you need to store it where your system can find it. This can be writing a file called `.env` and sticking in the API key. (If you already have a file like that, just add this as an extra line). The contents of your file should look like this:\n```\nOPENAI_API_KEY=sk-...\n```\n\nWhen you run this code, it will look for a `.env` file in the current directory; if it doesn't find it, it will look in the parent directory, and the parent's parent directory, and so on. This means that if you'd like, you can put the `.env` file in your home directory and code you execute will find it. If you're not running in a home directory, just make sure that there is a `.env` file in whatever directory you use.\n\nA word of warning: this key should stay secret. If you're working in a Git repo, be sure to .gitignore the .env file so you do not accidentally commit it.\n\n### Examples\nWe provide an example, but first, let's make some data. (Note that here our \"paragraph\" is only one or two words.)\n```python\n# Make some data that falls into natural classes\ndog_breeds = [\"Labrador Retriever\", \"German Shepherd\", \"Golden Retriever\", \"Bulldog\", \"Poodle\", \"Beagle\", \"Rottweiler\", \"Yorkshire Terrier\", \"Boxer\", \"Dachshund\"]\ncat_breeds = [\"Siamese\", \"Persian\", \"Maine Coon\", \"Ragdoll\", \"Bengal\", \"Sphynx\", \"British Shorthair\", \"Abyssinian\", \"Scottish Fold\", \"Russian Blue\"]\ncommon_birds = [\"Northern Cardinal\", \"American Robin\", \"Blue Jay\", \"House Sparrow\", \"Mourning Dove\", \"European Starling\", \"Black-capped Chickadee\", \"American Goldfinch\", \"Red-winged Blackbird\", \"Downy Woodpecker\"]\n\ndata = dog_breeds + cat_breeds + common_birds\nlabels = len(dog_breeds) * [\"dog\"] + len(cat_breeds) * [\"cat\"] + len(common_birds) * [\"birds\"]\n```\nNext, let's use an LLM to embed the data in a high dimension, then visualize it in two dimensions:\n```python\nimport mirabolic\n\nembedder = mirabolic.llm_embedder(X=data, L=labels)\nembedder.run_all()\n```\n\n![LLM_embedding_3](https://github.com/user-attachments/assets/d17b5967-a599-4969-81e6-12cef2ba7106)\n\nWow! Nice grouping of categories.\n\nSometimes we might want to process the text before embedding it. The `llm_embedder()` allows for arbitrary text transformations. Here's an example in which the transformation itself invokes another LLM.\n```python\nimport mirabolic\n\ndef f(s):\n \"\"\"\n Replace the breed or species with a pet name.\n \"\"\"\n prompt = (\n f\"I have a pet {s}. I need a name for my pet that is \"\n \"some sort of outrageous pun that will reveal its breed or species. \"\n \"Please reply with one name (and no other text).\"\n )\n response = mirabolic.LLM_API(prompt)\n print(f\"{s} => {response}\")\n return response\n\nembedder = mirabolic.llm_embedder(X=data, L=labels, f=f)\nembedder.run_all()\n```\nSo, we transform the species name into a cutesy pet name (e.g., `Boxer` -> `Rocky Bal-bow-wow`, or `Russian Blue` -> `Czar Purrtin`). Of course, this makes it more difficult to distinguish the underlying species, so the clusters become murkier. Nevertheless, the cat/dog/bird regions are still fairly visible:\n\n![LLM_embedding_2](https://github.com/user-attachments/assets/38c04782-5a58-4ed7-bf3c-8a1ec8d43476)\n\n\n## Comparing Event Rates\nSuppose you run a website. Every day you run a new email campaign to drive traffic to your site. You're considering a new approach, so you A/B test your campaigns: a portion of your emails use the new style of campaign and the rest use the old style.\n\nWhich style won? The new campaign looks good, but how do you know it's not just a random fluke? You vaguely remember an old stats teacher saying something about designing experiments with sufficient statistical power for a target effect size, but you don't know what the effect size is going to be, and you've already run the experiment! You really just want some simple way of visualizing the data you already have to see if the new style of campaigns are an improvement, or if it's all just random noise.\n\nWe provide such a tool here:\n```python\nimport numpy as np\nimport mirabolic\nimport matplotlib.pyplot as plt\n\n# Make some synthetic data\nnum_campaigns = 8\nnum_emails = 3000 # Number of emails in one arm of one campaign\n# In this example, the \"B\" arm has a slightly higher conversion rate\nnum_conversions_A = np.random.binomial(num_emails, 0.03, size=num_campaigns)\nnum_conversions_B = np.random.binomial(num_emails, 0.04, size=num_campaigns)\nnum_emails_A = num_campaigns * [num_emails]\nnum_emails_B = num_campaigns * [num_emails]\n\n# Plot the figure\nplt.figure(figsize=(6, 6))\nresults = mirabolic.rate_comparison(\n num_successes_A=num_conversions_A,\n num_successes_B=num_conversions_B,\n num_trials_A=num_emails_A,\n num_trials_B=num_emails_B,\n)\nplt.show()\n```\nHere's the output:\n\n![response_rate_1](https://github.com/user-attachments/assets/b43cdaca-7351-4e0a-b29f-73ced4d5a44c)\n\nThe figure shows a scatter plot with 8 points. Each point corresponds to a campaign, where the x-value is the conversion rate for the A arm (the old style, say) and the y-value is the conversion rate for the B arm (the new style). Around each point is a confidence rectangle showing how seriously to take it. If all the rectangles overlap with the diagonal line, then you don't have enough data to draw any conclusions (at least from individual campaigns). If the rectangles mostly fall above the dotted line, then the new style is an improvement; if below, it's making things worse.\n\n\n## CDFs with Confidence Intervals\n\nWhen exploring data, it can be very helpful to plot observations as a cumulative density function, or [CDF](https://en.wikipedia.org/wiki/Cumulative_distribution_function). (In medical contexts, doctors study the \"survival function\", which is 1 minus the CDF and essentially amounts to the same thing.) We plot a CDF by sorting the observed data from smallest to largest value. We can treat[^iid] the value in the middle of the sorted list as approximately the median, the value 90% of the way up the list as near the 90th percentile, and so forth.\n\n[^iid]: We assume the data consists of i.i.d. draws from some unknown probability distribution.\n\nWhen interpreting a CDF, or comparing two of them, one often wishes for something akin to a confidence interval. How close is the middle value to the median? Somewhat surprisingly, it is possible to compute the corresponding confidence intervals exactly.[^Beta]\n\n[^Beta]: More precisely, suppose we draw a sample of n i.i.d. observations and consider the i-th smallest; if we are sampling from *any* continuous probability distribution, then the distribution of the corresponding quantile has a [Beta distribution](https://en.wikipedia.org/wiki/Beta_distribution), B(i, n-i+1).\n\nFor a single data point, the uncertainty around its quantile can be thought of as a confidence interval. If we consider all the data points, then we refer to a *confidence band*.[^Credible]\n\n[^Credible]: Because we have access to a prior distribution on quantiles, these are arguably *[credible intervals](https://en.wikipedia.org/wiki/Credible_interval)* and *credible bands*, rather than confidence intervals and bands. We do not concern ourselves with this detail.\n\nWe provide a simple function for plotting CDFs with confidence bands; one invokes it by calling something like:\n```\nimport mirabolic\nimport matplotlib.pyplot as plt\nimport random\n\ndata = [random.random() for _ in range(50)]\nmirabolic.cdf_plot(data=data)\nplt.show()\n```\nThe figure looks like this:\n\n![CDF_1](https://github.com/user-attachments/assets/dbfb9640-78c4-4593-baf2-1ee1de591285)\n\nMore examples can be found in [`mirabolic/cdf/sample_usage.py`](https://github.com/Mirabolic/mirabolic/blob/main/mirabolic/cdf/sample_usage.py).\n\n## Neural Nets for GLM regression\n\nGLMs ([Generalized Linear Models](https://en.wikipedia.org/wiki/Generalized_linear_model)) are a relatively broad class of statistical model first popularlized in the 1970s. These have grown popular in the actuarial literature as a method of predicting insurance claims costs and frequency.\n\nWith the appropriate loss function, GLMs can be expressed as neural nets. These two techniques have traditionally been treated as distinct, but bridging the divide provides two advantages.\n\nFirst, a vast amount of effort has been spent on optimizing and accelerating neural nets over the past several years (GPUs and TPUs, parallelization). By expressing a GLM as a neural net, we can leverage this work.[^NN]\n\n[^NN]: In terms of focus, [this chart](https://trends.google.com/trends/explore?geo=US&q=deep%20learning,actuarial%20science) suggests something of the explosion of interest in neural nets and deep learning relative to more traditional actuarial models.\n\nSecond, expressing a GLM as a neural net opens the possibility of extending the neural net before or after the GLM component. For instance, suppose we build three subnets that each computed a single feature, and then feed the three outputs as inputs into the Poisson regression net. This single larger network would allow the three subnets to engineer their individual features such that the loss function of the joint network was optimized. This approach provides a straightforward way of performing non-linear feature engineering but retaining the explainability of a GLM. This two-step approach may provide regulatory advantages, since US Departments of Insurance (DOIs) have been reluctant to approve end-to-end deep learning models.\n\nWe provide loss functions for several of the most commonly used GLMs. Minimal code might look something like this:\n```\nimport mirabolic.neural_glm as neural_glm\nfrom keras.models import Sequential\nimport tf\n\nmodel = Sequential()\n# Actually design your neural net...\n# model.add(...)\nloss=neural_glm.Poisson_link_with_exposure\noptimizer = tf.keras.optimizers.Adam()\nmodel.compile(loss=neural_glm, optimizer=optimizer)\n```\n\nTo illustrate this process in more detail, we provide code to perform [Poisson regression](https://en.wikipedia.org/wiki/Poisson_regression) and Negative Binomial regression using a neural net. \n\nTo see the code in action, grab [the source code](https://github.com/Mirabolic/mirabolic) from GitHub, then [change to this directory](https://github.com/Mirabolic/mirabolic/tree/main/mirabolic/neural_glm), and run\n```\npython run_examples.py\n```\nThis will generate Poisson-distributed data and corresponding features and then try to recover the \"betas\" (i.e., the linear coefficients of the GLM) using various models, outputting both the true and recovered values.\n\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "Statistical and Machine Learning tools from Mirabolic",
"version": "0.3.10",
"project_urls": {
"Download": "https://github.com/Mirabolic/mirabolic/archive/refs/tags/v0.3.10.tar.gz",
"Homepage": "https://github.com/Mirabolic/mirabolic"
},
"split_keywords": [
"statistics",
" machine learning",
" cdf",
" quantiles"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "2dc4fd6f6e1ffb183f3ab7a245086cd3403d5113a5cf69e56fc9b2f64bc05742",
"md5": "07763e2c5ff2a8977446d4f4423f6b9b",
"sha256": "7fb0da0fbaf9a8309b451f4b9013202461ede020578371fa172ee3934468e322"
},
"downloads": -1,
"filename": "mirabolic-0.3.10-py3-none-any.whl",
"has_sig": false,
"md5_digest": "07763e2c5ff2a8977446d4f4423f6b9b",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": null,
"size": 34667,
"upload_time": "2024-11-28T14:27:36",
"upload_time_iso_8601": "2024-11-28T14:27:36.065520Z",
"url": "https://files.pythonhosted.org/packages/2d/c4/fd6f6e1ffb183f3ab7a245086cd3403d5113a5cf69e56fc9b2f64bc05742/mirabolic-0.3.10-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "87a41cd4895bfc5d8f9e9644b7c1d017f2d56f213f9fe794317787da022b2e0b",
"md5": "2714991da2f0193328c2bfec29a9125d",
"sha256": "355f1a149bcd48322327aa8568a9a6c7c415d9d3086d7c797070d333e2970587"
},
"downloads": -1,
"filename": "mirabolic-0.3.10.tar.gz",
"has_sig": false,
"md5_digest": "2714991da2f0193328c2bfec29a9125d",
"packagetype": "sdist",
"python_version": "source",
"requires_python": null,
"size": 35939,
"upload_time": "2024-11-28T14:27:37",
"upload_time_iso_8601": "2024-11-28T14:27:37.837704Z",
"url": "https://files.pythonhosted.org/packages/87/a4/1cd4895bfc5d8f9e9644b7c1d017f2d56f213f9fe794317787da022b2e0b/mirabolic-0.3.10.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-11-28 14:27:37",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "Mirabolic",
"github_project": "mirabolic",
"travis_ci": false,
"coveralls": false,
"github_actions": false,
"lcname": "mirabolic"
}