ezbolt


Nameezbolt JSON
Version 0.2.0 PyPI version JSON
download
home_pageNone
Summaryezbolt - bolt force calculations in python
upload_time2024-12-08 04:49:56
maintainerNone
docs_urlNone
authorNone
requires_python>=3.7
licenseNone
keywords
VCS
bugtrack_url
requirements contourpy cycler fonttools kiwisolver matplotlib numpy packaging pandas Pillow pyparsing python-dateutil pytz six tzdata
Travis-CI No Travis.
coveralls test coverage No coveralls.
            <h1 align="center">
  <br>
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/logo.png?raw=true" alt="logo" style="width: 60%;" />
  <br>
  Bolt Force Calculation in Python
  <br>
</h1>
<p align="center">
Calculate bolt forces with Elastic Method and Instant Center of Rotation (ICR) method.
</p>

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/demo.gif?raw=true" alt="demo" style="width: 75%;" />
</div>


- [Introduction](#introduction)
- [Quick Start](#quick-start)
- [Installation](#installation)
- [Usage](#usage)
- [Theoretical Background - Elastic Method](#theoretical-background---elastic-method)
- [Theoretical Background - ICR Method](#theoretical-background---icr-method)
- [Theoretical Background - Brandt's Method for Locating ICR](#theoretical-background---brandt-s-method-for-locating-icr)
- [Assumptions and Limitations](#assumptions-and-limitations)
- [License](#license)




## Introduction

EZbolt is a Python program that calculates bolt forces in a bolt group subject to shear and in-plane torsion. It does so using both the Elastic Method and the Instant Center of Rotation (ICR) method as outlined in the AISC steel construction manual. The iterative algorithm for locating the center of rotation is explained in this paper by Donald Brandt: [Rapid Determination of Ultimate Strength of Eccentrically Loaded Bolt Groups.](https://www.aisc.org/Rapid-Determination-of-Ultimate-Strength-of-Eccentrically-Loaded-Bolt-Groups). Unlike the ICR coefficient tables in the steel construction manual which is provided in 15 degree increments, EZbolt can handle **any bolt arrangements, any load orientation, and any eccentricity**.

> [!TIP]
>
> Don't have python experience? Worry not, you will find a .csv file in the `Cu Coefficient Table` folder. 90,000 common bolt configurations have been pre-computed and tabulated. Need to find some Cu coefficient for your connection design? Just copy the csv into your spreadsheet and do some VLOOKUP. No solvers needed!



### Tabulated Cu Coefficients Table

* `Cu Coefficient.csv`
  * **columns**: column of bolts 
  * **rows**: row of bolts 
  * **eccentricity**: load eccentricity (ex = Mz / Vy) 
  * **degree**:  load orientation (0 degrees is vertical downward) 
  * **Ce**: elastic center of rotation coefficient
  * **Cu**: (plastic) instant center of rotation coefficient
* `Cu Coefficient.json`
  * This is a nested dictionary for engineers more comfortable with python. Dictionary lookup is pretty much instant whereas solving a bolt configuration may take ~ 100 ms depending on your computer. 
  * The key order is as follows: `...[N_columns][N_rows][eccentricity][degree]`["Cu" or "Ce"]. All keys are integers.
  * For example, `...[1][6][6][0]["Cu"]` returns the Cu for a single column of bolt with 6 rows, vertical force with 6" eccentricity. The returned Cu is 3.55 which matches the AISC tables.

If you would like to generate your own table, try running `generate_cu_table.py`. You can specify the range for each parameter. Be careful though, the number of configurations increase exponentially. Also there's tricky convergence issues if e<0.5 and degree > 80. The cached coefficients have the following range:

* **columns**: 1 to 3
* **rows**: 2 to 12
* **eccentricity**: 1 to 36
* **degree**: 0 to 75

That's 3 * 11 * 76 * 36 = 90,288 iterations. On my Linux desktop with an Intel i7-11700, each iteration took ~ 50 ms. A serial run would take ~75 minutes. Luckily, I was able to implement some nifty parallel processing to bring that the run time to ~5 minutes (running 16 threads).




## Quick Start

Run main.py:

```python
import ezbolt

# initialize a bolt group
bolt_group = ezbolt.BoltGroup()

# add a 3x3 bolt group with 6" width and 6" depth with lower left corner located at (0,0)
bolt_group.add_bolts(xo=0, yo=0, width=6, height=6, nx=3, ny=3)

# preview geometry
ezbolt.plotter.preview(bolt_group)

# calculate bolt demands under 50 kips horizontal shear, 50 kips vertical shear, and 200 k.in torsion
results = bolt_group.solve(Vx=50, Vy=50, torsion=200, bolt_capacity=17.9)

# plot bolt forces
ezbolt.plot_elastic(bolt_group)
ezbolt.plot_ECR(bolt_group)
ezbolt.plot_ICR(bolt_group)

# look at the bolt force tables
df1 = results["Elastic Method - Superposition"]["Bolt Force Table"]
df2 = results["Elastic Method - Center of Rotation"]["Bolt Force Table"]
df3 = results["Instant Center of Rotation Method"]["Bolt Force Table"]
```

`ezbolt.preview()` plots a bolt group preview:

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/preview.png?raw=true" alt="demo" style="width: 50%;" />
</div>
`ezbolt.plot_elastic()` shows bolt force calculated from elastic method.

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/elasticmethod.png?raw=true" alt="demo" style="width: 50%;" />
</div>
`ezbolt.plot_ECR()` shows bolt forces calculated from elastic center of rotation (ECR) method.

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/ECRmethod.png?raw=true" alt="demo" style="width: 50%;" />
</div>
`ezbolt.plot_ICR()` shows bolt forces calculated from instant center of rotation (ICR) method.

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/ICRmethod.png?raw=true" alt="demo" style="width: 50%;" />
</div>

`BoltGroup.solve()` returns a dictionary containing all relevant calculation results:

* `results["Elastic Method - Superposition"]`
    * `... ["Bolt Capacity"]`
    * `... ["Bolt Demand"]`
    * `... ["Bolt Force Table"]`
    * `... ["DCR"]`
* `results["Elastic Method - Center of Rotation"]`
    * `... ["Center of Rotation"]`
    * `... ["Ce"]`
    * `... ["Connection Capacity"]`
    * `... ["Connection Demand"]`
    * `... ["Bolt Force Table"]`
    * `... ["DCR"]`
* `results["Instant Center of Rotation Method"]`
    * `... ["ICR"]`
    * `... ["Cu"]`
    * `... ["Connection Capacity"]`
    * `... ["Connection Demand"]`
    * `... ["Bolt Force Table"]`
    * `... ["DCR"]`


## Installation

**Option 1: Anaconda Python**

Simply run main.py using the default Anaconda base environment. The following packages are required:

* Numpy
* Matplotlib
* Pandas

Installation procedure:

1. Download Anaconda python
2. Download this package (click the green "Code" button and download zip file)
3. Open and run "main.py" in Anaconda's Spyder IDE.

**Option 2: Standalone Python**

1. Download this project to a folder of your choosing
    ```
    git clone https://github.com/wcfrobert/ezbolt.git
    ```
2. Change directory into where you downloaded ezbolt
    ```
    cd ezbolt
    ```
3. Create virtual environment
    ```
    py -m venv venv
    ```
4. Activate virtual environment
    ```
    venv\Scripts\activate
    ```
5. Install requirements
    ```
    pip install -r requirements.txt
    ```
6. run ezbolt
    ```
    py main.py
    ```
    Pip install is available:

```
pip install ezbolt
```

## Usage

Here are all the public methods available to the user:

**Adding Bolts**

* `ezbolt.BoltGroup.add_bolts(xo, yo, width, height, nx, ny, perimeter_only=False)`
* `ezbolt.BoltGroup.add_bolt_single(x, y)`

**Solving**

* `ezbolt.BoltGroup.solve(Vx, Vy, torsion, bolt_capacity=17.9, verbose=True, ecc_method="AISC")`

**Visualizations**

* `ezbolt.preview(boltgroup_object)`
* `ezbolt.plot_elastic(boltgroup_object, annotate_force=True)`
* `ezbolt.plot_ECR(boltgroup_object, annotate_force=True)`
* `ezbolt.plot_ICR(boltgroup_object, annotate_force=True)`

For further guidance and documentation, you can access the docstring of any method using the help() command. For example, here is the output for `help(ezbolt.BoltGroup.solve)`

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/help.png?raw=true" alt="demo" style="width: 90%;" />
</div>



## Theoretical Background - Elastic Method

A group of bolts can be treated like any geometric section, and their geometric properties can be calculated (e.g. centroid, moment of inertia, etc):

Centroid:

$$x_{cg} = \frac{\sum x_i}{N_{bolts}}$$

$$y_{cg} = \frac{\sum y_i}{N_{bolts}}$$

Moment of inertia about x and y axis:

$$I_x = \sum (y_i - y_{cg})^2$$

$$I_y = \sum (x_i - x_{cg})^2$$

Polar moment of inertia:

$$I_z = J = I_p = I_x + I_y$$

For in-plane shear, the resulting demand on individual bolts is simply total force divided by number of bolts. We do this about the x and y components separately. Let's call this **direct shear**.

$$v_{dx} = \frac{V_x}{N_{bolts}}$$

$$v_{dy} = \frac{V_y}{N_{bolts}}$$


In-plane torsion on the bolt group is converted to shear on the individual anchors. Let's call this **torsional shear**. The equations below should be very familiar to most engineers. They're identical to the torsion shear stress equations for beam sections ($$\tau = Tc/J$$). Essentially, bolt force varies linearly radiating from the centroid. Bolts furthest away from the centroid naturally take more force.

$$v_{tx} = \frac{M_z (y_i - y_{cg})}{I_z}$$

$$v_{ty} = \frac{-M_z (x_i - x_{cg})}{I_z}$$

Putting it all together, we use principle of superposition to **superimpose** (add) the bolt demands together. In other words, we can look at each action separately, then add them together at the end.

$$v_{x} = v_{dx} + v_{tx}$$

$$v_{y} = v_{dy} + v_{ty}$$

$$v_{resultant} = \sqrt{v_{x}^2 + v_{y}^2}$$

The key assumption of elastic method is that rotational and translational actions are decoupled and do not influence each other. This is not true and produces conservative results.


## Theoretical Background - ICR Method

A less conservative way of determining bolt forces is through the Instant Center of Rotation **(ICR)** method. The underlying theory is illustrated in the figure below. In short, when a bolt group is subjected to combined in-plane force and torsion, the bolt force and applied force vectors **revolve around an imaginary center**. The ICR method is conceptually simple, but identifying the ICR location will require iteration.

Rather than assuming elasticity and using geometric properties to determine bolt forces, we assume the bolt furthest from ICR has a deformation of 0.34", other bolts are assumed to have linearly varying deformation based on its distance from ICR (varying from 0" to 0.34"). We can then determine the corresponding force using the **force-deformation relationship** below.

$$\Delta_i = 0.34 (d_i / d_{max})$$

$$R_i =(1-e^{-10\Delta_i})^{0.55} \times R_{max}$$

Assuming the location of ICR has been correctly identified, equilibrium should hold.

$$\sum F_x = 0 = P_x -\sum R_{ix}$$

$$\sum F_y = 0 = P_y -\sum R_{iy}$$

$$\sum M = 0 = M_z -\sum m_i$$

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/aisc2.31.png?raw=true" alt="demo" style="width: 60%;" />
</div>
Compared to the elastic method, the ICR method differs in four key ways:

* The **applied load vectors (Vx, Vy, Mz) is converted into an equivalent eccentricity and load orientation** 

$$(V_x,V_y,M_z) \rightarrow (P, e_x,\theta)$$

* Rather than looking at DCR on the individual bolt level, ICR method provides an overall connection capacity. A coefficient (C) is determined, and we can convert bolt capacity to the overall connection capacity as follows.

$$\mbox{connection capacity} = C \times \mbox{bolt capacity}$$

* ICR method is more accurate and less conservative because it allows for **plastic deformation of bolts**. An appropriate analogy would be how the plastic section modulus (Zx) is larger than elastic section modulus (Sx). Technically, the elastic method also has a center about which bolt force vectors revolve. But because everything is linear, we can skip the force-deformation relationship and calculate forces from geometric properties like moment of inertia and polar moment of inertia.
* Unlike the elastic method, **ICR method is not practical to do by hand** as the location of ICR must be determined iteratively. There are design tables available in the steel construction manual. Generally the ICR method is more of a black-box.

The derivations will follow AISC notations which is somewhat different from what I've used above. Most notably, we will use **P** to denote applied force instead of **V**, and **R** to denote in-plane bolt force instead of **v**.

Suppose we know the exact location of ICR, first let's calculate the applied moment with respect to this new center.

$$M_p = P \times r_o$$

The maximum bolt deformation of 0.34" occurs at the bolt furthest from ICR, the other bolts have deformation varying linearly between 0 to 0.34 based on its distance to the ICR ($d_i$)

$$\Delta_{max} = 0.34$$

$$\Delta_i = 0.34 \frac{d_i}{d_{max}}$$

Next, we can calculate individual bolt forces using the following force-deformation relationship:

$$R_i = (1-e^{-10\Delta_i})^{0.55} \times R_{max}$$

Now, we can calculate the reactive moment contributions from each bolt:

$$M_i = R_i \times d_i$$

$$M_i = R_{max} (1-e^{-10\Delta_i})^{0.55} \times d_i$$

$$\sum M_i = R_{max} \times \sum (1-e^{-10\Delta_i})^{0.55} d_i$$

The applied moment, and the reactive moment are in equilibrium. Rearrange for P:

$$ M_{applied} = M_{resisting}$$

$$ M_p = \sum M_i$$

$$ P \times r_o = R_{max} \times \sum (1-e^{-10\Delta_i})^{0.55} d_i$$

$$ P = R_{max} \times \frac{\sum (1-e^{-10\Delta_i})^{0.55} d_i}{r_o}$$

Let the second term be the ICR coefficient C. **You can think of C as a constant that converts an applied load (P) to the maximum bolt demand**

$$ P = R_{max} \times C$$

$$ C = \frac{\sum (1-e^{-10\Delta_i})^{0.55} d_i}{r_o}$$

Set $R_{max}$ equal to the bolt capacity and back-calculate the connection capacity:

$$ R_{max} = R_{capacity}$$

$$ P_{capacity} = R_{capacity} \times C$$

Notes:

1. Despite a nonlinear bolt force-deformation, the relationship between max bolt force ($R_{max}$) and applied force ($P$) is linear. In other words, if applied force doubles, so does maximum bolt force, and vice versa. Embedded in this is the **assumption that eccentricity (e = Mz / P) will remain constant**      
2. If we substitute 0.34 into the exponential function above, we get 0.9815 as there's a horizontal asymptote and we will never reach 1.0 exactly. We can make a simple adjustment to our "C" equation if we desire. Note that AISC does NOT make this adjustment as it is more conservative to set max bolt-force as $0.9815R_{max}$, effectively capping our utilization ratio to 98%.

$$ (1 - e^{-10(0.34)} )^{0.55} = 0.9815 $$

$$ C = \frac{\sum (1-e^{-10\Delta_i})^{0.55} d_i}{0.9815 r_o}$$


## Theoretical Background - Brandt's Method for Locating ICR

The derivation above assumes we know where ICR is. But we don't, and it is not a trivial task to find it. The [original Crawford and Kulak paper (1971)](https://ascelibrary.org/doi/10.1061/JSDEAG.0002844) is somewhat misleading. The search space for ICR is very rarely a one-dimensional line except in the very specific situation where load angle is 0 degrees. In other words, you cannot draw an orthogonal line from P to CoG, extend that line, and expect to find ICR somewhere along it. This is explained in detail by [Muir and Thornton (2004)](https://www.cives.com/cives-engineering-corporation-publications); the search space for ICR is almost always two-dimensional.

Luckily for us, there exists an iterative method that converges on ICR very quickly. [Brandt's method (1982)](https://www.aisc.org/Rapid-Determination-of-Ultimate-Strength-of-Eccentrically-Loaded-Bolt-Groups) is fast and efficient, and it is what AISC uses to construct their design tables. We will implement Brandt's method here. The two key insights presented by Brandt is summarized below:

**Insight #1: Elastic method also has a center of rotation and can be readily calculated**

Let $$(x_{cg}, y_{cg})$$ be the coordinate of bolt group centroid, the coordinate for the elastic center of rotation (ECR) is $$(x_{cg} +a_x,  y_{cg}+a_y)$$:

$$a_x = \frac{P_y}{n} \frac{J}{M_z}$$

$$a_y = \frac{P_x}{n} \frac{J}{M_z}$$

Where:

* $$P_x$$ = x component of applied load
* $$P_y$$= y component of applied load
* $$n$$ = number of bolts in bolt group
* $$J$$ = polar moment of inertia ($$I_z$$)
* $$M_z$$ = applied torsion

**Insight #2: Elastic center of rotation (ECR) can be used as the initial guess of (ICR), subsequent improvements can be achieved as follows:**

Let the initial guess be the ECR:

$$x_0 = x_{cg} +a_x$$

$$y_0 = y_{cg} + a_y$$

At this assumed ICR location, calculate force equilibrium. Note how moment equilibrium is enforced when we determine $$R_{max}$$ at each step.

$$\sum F_x = f_{xx} = P_x - \sum R_{x}$$

$$\sum F_y = f_{yy} = P_y - \sum R_{y}$$

The force summations will not be zero unless we are at the ICR, use the residual to determine successive guesses:

$$x_{i+1} = x_i - \frac{f_{yy}J}{nM_z}$$

$$y_{i+1} = y_i + \frac{f_{xx}J}{nM_z}$$

Repeat until the desired tolerance is achieved:

$$\mbox{residual} = \sqrt{(f_{xx})^2 + (f_{yy})^2} < tol$$



Here is the step by step procedure:


* **Step 1**: From applied force $(P_x, P_y, M_z)$, calculate load vector orientation ($\theta$). Note [atan2](https://en.wikipedia.org/wiki/Atan2) is a specialized arctan function that returns within the range between -180 to 180 degrees, rather than -90 to 90 degrees. This is to obtain a correct and unambiguous value for the angle theta.

$$P = \sqrt{P_x^2 + P_y^2}$$

$$\theta = atan2(\frac{P_y}{P_x})$$

* **Step 2**: Now calculate eccentricity and its x and y components. $e_x$ and $e_y$ is used to locate the point of applied load (let's call this point P). The location of P is actually ambiguous and can be anywhere along $L(x) = P_ye_x - P_xe_y$. One method is to place P such that the line P-CoG is perpendicular to L(x). Another alternative is to assume $e_y=0$ which is what AISC assumes and what we will do by default. Note both methods lead to the same result as long as P is oriented along L(x). You can change which assumption is used by ezbolt by changing the `ecc_method` argument in `ezbolt.BoltGroup.solve()`

$$e_y = 0$$

$$e_x = M_z/P_y$$

$$e = \sqrt{e_x^2 + e_y^2}$$

* **Step 3**: Obtain an initial guess of ICR location per Brandt's method, then calculate distance of line P-ICR ($r_o$)

$$a_x = P_y \times \frac{I_z}{M_z N_{bolt}}$$

$$a_y = P_x \times \frac{I_z}{M_z N_{bolt}}$$

$$x_{ICR} = x_{cg} - a_x$$

$$y_{ICR} = y_{cg} + a_y$$

$$r_{ox} = e_x + a_x$$

$$r_{oy} = e_y - a_y$$

$$r_{o} = \sqrt{r_{ox}^2 + r_{oy}^2}$$

* **Step 4**: At this point, we can already compute coefficient "C" at our assumed ICR location.

$$C = \frac{ \sum((1 - e^{-10 \Delta_i})^{0.55} d_i)}{r_o}$$
    
* **Step 5**: Next, we need to determine the maximum bolt force ($R_{max}$) at the user-specified load magnitude. This can be done by using the moment equilibrium equation; hence why we only need to check force equilibrium at the end. Moment equilibrium is established as a matter of course by enforcing a specific value of $R_{max}$

$$M_p = P_x r_{oy} - P_y  r_{ox}$$

$$M_r = R_{max} \times \sum((1 - e^{-10 \Delta_i})^{0.55} d_i)$$

$$R_{max} = \frac{M_p}{\sum((1 - e^{-10 \Delta_i})^{0.55} d_i)}$$

* **Step 6**: Now that we have $R_{max}$, we can calculate the other bolt forces:

$$R_i = R_{max} \times \sum((1 - e^{-10 \Delta_i})^{0.55} d_i)$$

* **Step 7**: Now calculate the bolt forces' x and y component to check force equilibrium:

$$cos(\theta) = d_x / d = sin(\theta+90^o)$$

$$sin(\theta) = d_y / d = -cos(\theta+90^o)$$

$$R_x = R_i cos(\theta+90^o) = -R_i \frac{d_y}{d}$$

$$R_y = R_i sin(\theta+90^o) = R_i \frac{d_x}{d}$$

* **Step 8**: Calculate residual and repeat until a specific tolerance is achieved.

$$\sum F_x = f_{xx} = P_x - \sum R_{x}$$

$$\sum F_y = f_{yy} = P_y - \sum R_{y}$$

$$\mbox{residual} = \sqrt{(f_{xx})^2 + (f_{yy})^2} < tol$$

* **Step 9**: If equilibrium is not achieved, go back to step 4 with the following modifications

$$a_x = f_{yy} \times \frac{I_z}{M_z N_{bolt}}$$

$$a_y = f_{xx} \times \frac{I_z}{M_z N_{bolt}}$$

$$x_{ICR,i} = x_{ICR,i-1} - a_x$$

$$y_{ICR,i} = x_{ICR,i-1} + a_y$$

* **Step 10**: Once ICR has been located, calculate connection capacity:

$$P_{capacity} = C \times R_{capacity}$$

$$DCR = \frac{P}{P_{capacity}} = \frac{R_{max}}{R_{capacity}}$$

Easy enough to implement. The hard part is making sure you make an even number of sign errors.






## Assumptions and Limitations

* Sign convention follows the right-hand rule. right is +X, top is +Y, counter-clockwise is positive torsion. Note that since we are only concerned with in-plane forces, only the highlighted vectors are relevant.

<div align="center">
  <img src="https://github.com/wcfrobert/ezbolt/blob/master/doc/signconvention.png?raw=true" alt="demo" style="width: 75%;" />
</div>

* Units are in (kip, in) unless otherwise noted
* EZbolt only calculates connection capacity with respect to bolt shear. Other limit states - such as plate rupture, block shear, bearing and tearout - are not considered.



## License

MIT License

Copyright (c) 2023 Robert Wang
            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "ezbolt",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": null,
    "keywords": null,
    "author": null,
    "author_email": "wcfrobert <wcfrobert@hotmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/77/17/262535959558462b08e5dbed69a256504db3e5fc74bd948981cc1fbf732e/ezbolt-0.2.0.tar.gz",
    "platform": null,
    "description": "<h1 align=\"center\">\n  <br>\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/logo.png?raw=true\" alt=\"logo\" style=\"width: 60%;\" />\n  <br>\n  Bolt Force Calculation in Python\n  <br>\n</h1>\n<p align=\"center\">\nCalculate bolt forces with Elastic Method and Instant Center of Rotation (ICR) method.\n</p>\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/demo.gif?raw=true\" alt=\"demo\" style=\"width: 75%;\" />\n</div>\n\n\n- [Introduction](#introduction)\n- [Quick Start](#quick-start)\n- [Installation](#installation)\n- [Usage](#usage)\n- [Theoretical Background - Elastic Method](#theoretical-background---elastic-method)\n- [Theoretical Background - ICR Method](#theoretical-background---icr-method)\n- [Theoretical Background - Brandt's Method for Locating ICR](#theoretical-background---brandt-s-method-for-locating-icr)\n- [Assumptions and Limitations](#assumptions-and-limitations)\n- [License](#license)\n\n\n\n\n## Introduction\n\nEZbolt is a Python program that calculates bolt forces in a bolt group subject to shear and in-plane torsion. It does so using both the Elastic Method and the Instant Center of Rotation (ICR) method as outlined in the AISC steel construction manual. The iterative algorithm for locating the center of rotation is explained in this paper by Donald Brandt: [Rapid Determination of Ultimate Strength of Eccentrically Loaded Bolt Groups.](https://www.aisc.org/Rapid-Determination-of-Ultimate-Strength-of-Eccentrically-Loaded-Bolt-Groups). Unlike the ICR coefficient tables in the steel construction manual which is provided in 15 degree increments, EZbolt can handle **any bolt arrangements, any load orientation, and any eccentricity**.\n\n> [!TIP]\n>\n> Don't have python experience? Worry not, you will find a .csv file in the `Cu Coefficient Table` folder. 90,000 common bolt configurations have been pre-computed and tabulated. Need to find some Cu coefficient for your connection design? Just copy the csv into your spreadsheet and do some VLOOKUP. No solvers needed!\n\n\n\n### Tabulated Cu Coefficients Table\n\n* `Cu Coefficient.csv`\n  * **columns**: column of bolts \n  * **rows**: row of bolts \n  * **eccentricity**: load eccentricity (ex = Mz / Vy) \n  * **degree**:  load orientation (0 degrees is vertical downward) \n  * **Ce**: elastic center of rotation coefficient\n  * **Cu**: (plastic) instant center of rotation coefficient\n* `Cu Coefficient.json`\n  * This is a nested dictionary for engineers more comfortable with python. Dictionary lookup is pretty much instant whereas solving a bolt configuration may take ~ 100 ms depending on your computer. \n  * The key order is as follows: `...[N_columns][N_rows][eccentricity][degree]`[\"Cu\" or \"Ce\"]. All keys are integers.\n  * For example, `...[1][6][6][0][\"Cu\"]` returns the Cu for a single column of bolt with 6 rows, vertical force with 6\" eccentricity. The returned Cu is 3.55 which matches the AISC tables.\n\nIf you would like to generate your own table, try running `generate_cu_table.py`. You can specify the range for each parameter. Be careful though, the number of configurations increase exponentially. Also there's tricky convergence issues if e<0.5 and degree > 80. The cached coefficients have the following range:\n\n* **columns**: 1 to 3\n* **rows**: 2 to 12\n* **eccentricity**: 1 to 36\n* **degree**: 0 to 75\n\nThat's 3 * 11 * 76 * 36 = 90,288 iterations. On my Linux desktop with an Intel i7-11700, each iteration took ~ 50 ms. A serial run would take ~75 minutes. Luckily, I was able to implement some nifty parallel processing to bring that the run time to ~5 minutes (running 16 threads).\n\n\n\n\n## Quick Start\n\nRun main.py:\n\n```python\nimport ezbolt\n\n# initialize a bolt group\nbolt_group = ezbolt.BoltGroup()\n\n# add a 3x3 bolt group with 6\" width and 6\" depth with lower left corner located at (0,0)\nbolt_group.add_bolts(xo=0, yo=0, width=6, height=6, nx=3, ny=3)\n\n# preview geometry\nezbolt.plotter.preview(bolt_group)\n\n# calculate bolt demands under 50 kips horizontal shear, 50 kips vertical shear, and 200 k.in torsion\nresults = bolt_group.solve(Vx=50, Vy=50, torsion=200, bolt_capacity=17.9)\n\n# plot bolt forces\nezbolt.plot_elastic(bolt_group)\nezbolt.plot_ECR(bolt_group)\nezbolt.plot_ICR(bolt_group)\n\n# look at the bolt force tables\ndf1 = results[\"Elastic Method - Superposition\"][\"Bolt Force Table\"]\ndf2 = results[\"Elastic Method - Center of Rotation\"][\"Bolt Force Table\"]\ndf3 = results[\"Instant Center of Rotation Method\"][\"Bolt Force Table\"]\n```\n\n`ezbolt.preview()` plots a bolt group preview:\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/preview.png?raw=true\" alt=\"demo\" style=\"width: 50%;\" />\n</div>\n`ezbolt.plot_elastic()` shows bolt force calculated from elastic method.\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/elasticmethod.png?raw=true\" alt=\"demo\" style=\"width: 50%;\" />\n</div>\n`ezbolt.plot_ECR()` shows bolt forces calculated from elastic center of rotation (ECR) method.\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/ECRmethod.png?raw=true\" alt=\"demo\" style=\"width: 50%;\" />\n</div>\n`ezbolt.plot_ICR()` shows bolt forces calculated from instant center of rotation (ICR) method.\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/ICRmethod.png?raw=true\" alt=\"demo\" style=\"width: 50%;\" />\n</div>\n\n`BoltGroup.solve()` returns a dictionary containing all relevant calculation results:\n\n* `results[\"Elastic Method - Superposition\"]`\n    * `... [\"Bolt Capacity\"]`\n    * `... [\"Bolt Demand\"]`\n    * `... [\"Bolt Force Table\"]`\n    * `... [\"DCR\"]`\n* `results[\"Elastic Method - Center of Rotation\"]`\n    * `... [\"Center of Rotation\"]`\n    * `... [\"Ce\"]`\n    * `... [\"Connection Capacity\"]`\n    * `... [\"Connection Demand\"]`\n    * `... [\"Bolt Force Table\"]`\n    * `... [\"DCR\"]`\n* `results[\"Instant Center of Rotation Method\"]`\n    * `... [\"ICR\"]`\n    * `... [\"Cu\"]`\n    * `... [\"Connection Capacity\"]`\n    * `... [\"Connection Demand\"]`\n    * `... [\"Bolt Force Table\"]`\n    * `... [\"DCR\"]`\n\n\n## Installation\n\n**Option 1: Anaconda Python**\n\nSimply run main.py using the default Anaconda base environment. The following packages are required:\n\n* Numpy\n* Matplotlib\n* Pandas\n\nInstallation procedure:\n\n1. Download Anaconda python\n2. Download this package (click the green \"Code\" button and download zip file)\n3. Open and run \"main.py\" in Anaconda's Spyder IDE.\n\n**Option 2: Standalone Python**\n\n1. Download this project to a folder of your choosing\n    ```\n    git clone https://github.com/wcfrobert/ezbolt.git\n    ```\n2. Change directory into where you downloaded ezbolt\n    ```\n    cd ezbolt\n    ```\n3. Create virtual environment\n    ```\n    py -m venv venv\n    ```\n4. Activate virtual environment\n    ```\n    venv\\Scripts\\activate\n    ```\n5. Install requirements\n    ```\n    pip install -r requirements.txt\n    ```\n6. run ezbolt\n    ```\n    py main.py\n    ```\n    Pip install is available:\n\n```\npip install ezbolt\n```\n\n## Usage\n\nHere are all the public methods available to the user:\n\n**Adding Bolts**\n\n* `ezbolt.BoltGroup.add_bolts(xo, yo, width, height, nx, ny, perimeter_only=False)`\n* `ezbolt.BoltGroup.add_bolt_single(x, y)`\n\n**Solving**\n\n* `ezbolt.BoltGroup.solve(Vx, Vy, torsion, bolt_capacity=17.9, verbose=True, ecc_method=\"AISC\")`\n\n**Visualizations**\n\n* `ezbolt.preview(boltgroup_object)`\n* `ezbolt.plot_elastic(boltgroup_object, annotate_force=True)`\n* `ezbolt.plot_ECR(boltgroup_object, annotate_force=True)`\n* `ezbolt.plot_ICR(boltgroup_object, annotate_force=True)`\n\nFor further guidance and documentation, you can access the docstring of any method using the help() command. For example, here is the output for `help(ezbolt.BoltGroup.solve)`\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/help.png?raw=true\" alt=\"demo\" style=\"width: 90%;\" />\n</div>\n\n\n\n## Theoretical Background - Elastic Method\n\nA group of bolts can be treated like any geometric section, and their geometric properties can be calculated (e.g. centroid, moment of inertia, etc):\n\nCentroid:\n\n$$x_{cg} = \\frac{\\sum x_i}{N_{bolts}}$$\n\n$$y_{cg} = \\frac{\\sum y_i}{N_{bolts}}$$\n\nMoment of inertia about x and y axis:\n\n$$I_x = \\sum (y_i - y_{cg})^2$$\n\n$$I_y = \\sum (x_i - x_{cg})^2$$\n\nPolar moment of inertia:\n\n$$I_z = J = I_p = I_x + I_y$$\n\nFor in-plane shear, the resulting demand on individual bolts is simply total force divided by number of bolts. We do this about the x and y components separately. Let's call this **direct shear**.\n\n$$v_{dx} = \\frac{V_x}{N_{bolts}}$$\n\n$$v_{dy} = \\frac{V_y}{N_{bolts}}$$\n\n\nIn-plane torsion on the bolt group is converted to shear on the individual anchors. Let's call this **torsional shear**. The equations below should be very familiar to most engineers. They're identical to the torsion shear stress equations for beam sections ($$\\tau = Tc/J$$). Essentially, bolt force varies linearly radiating from the centroid. Bolts furthest away from the centroid naturally take more force.\n\n$$v_{tx} = \\frac{M_z (y_i - y_{cg})}{I_z}$$\n\n$$v_{ty} = \\frac{-M_z (x_i - x_{cg})}{I_z}$$\n\nPutting it all together, we use principle of superposition to **superimpose** (add) the bolt demands together. In other words, we can look at each action separately, then add them together at the end.\n\n$$v_{x} = v_{dx} + v_{tx}$$\n\n$$v_{y} = v_{dy} + v_{ty}$$\n\n$$v_{resultant} = \\sqrt{v_{x}^2 + v_{y}^2}$$\n\nThe key assumption of elastic method is that rotational and translational actions are decoupled and do not influence each other. This is not true and produces conservative results.\n\n\n## Theoretical Background - ICR Method\n\nA less conservative way of determining bolt forces is through the Instant Center of Rotation **(ICR)** method. The underlying theory is illustrated in the figure below. In short, when a bolt group is subjected to combined in-plane force and torsion, the bolt force and applied force vectors **revolve around an imaginary center**. The ICR method is conceptually simple, but identifying the ICR location will require iteration.\n\nRather than assuming elasticity and using geometric properties to determine bolt forces, we assume the bolt furthest from ICR has a deformation of 0.34\", other bolts are assumed to have linearly varying deformation based on its distance from ICR (varying from 0\" to 0.34\"). We can then determine the corresponding force using the **force-deformation relationship** below.\n\n$$\\Delta_i = 0.34 (d_i / d_{max})$$\n\n$$R_i =(1-e^{-10\\Delta_i})^{0.55} \\times R_{max}$$\n\nAssuming the location of ICR has been correctly identified, equilibrium should hold.\n\n$$\\sum F_x = 0 = P_x -\\sum R_{ix}$$\n\n$$\\sum F_y = 0 = P_y -\\sum R_{iy}$$\n\n$$\\sum M = 0 = M_z -\\sum m_i$$\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/aisc2.31.png?raw=true\" alt=\"demo\" style=\"width: 60%;\" />\n</div>\nCompared to the elastic method, the ICR method differs in four key ways:\n\n* The **applied load vectors (Vx, Vy, Mz) is converted into an equivalent eccentricity and load orientation** \n\n$$(V_x,V_y,M_z) \\rightarrow (P, e_x,\\theta)$$\n\n* Rather than looking at DCR on the individual bolt level, ICR method provides an overall connection capacity. A coefficient (C) is determined, and we can convert bolt capacity to the overall connection capacity as follows.\n\n$$\\mbox{connection capacity} = C \\times \\mbox{bolt capacity}$$\n\n* ICR method is more accurate and less conservative because it allows for **plastic deformation of bolts**. An appropriate analogy would be how the plastic section modulus (Zx) is larger than elastic section modulus (Sx). Technically, the elastic method also has a center about which bolt force vectors revolve. But because everything is linear, we can skip the force-deformation relationship and calculate forces from geometric properties like moment of inertia and polar moment of inertia.\n* Unlike the elastic method, **ICR method is not practical to do by hand** as the location of ICR must be determined iteratively. There are design tables available in the steel construction manual. Generally the ICR method is more of a black-box.\n\nThe derivations will follow AISC notations which is somewhat different from what I've used above. Most notably, we will use **P** to denote applied force instead of **V**, and **R** to denote in-plane bolt force instead of **v**.\n\nSuppose we know the exact location of ICR, first let's calculate the applied moment with respect to this new center.\n\n$$M_p = P \\times r_o$$\n\nThe maximum bolt deformation of 0.34\" occurs at the bolt furthest from ICR, the other bolts have deformation varying linearly between 0 to 0.34 based on its distance to the ICR ($d_i$)\n\n$$\\Delta_{max} = 0.34$$\n\n$$\\Delta_i = 0.34 \\frac{d_i}{d_{max}}$$\n\nNext, we can calculate individual bolt forces using the following force-deformation relationship:\n\n$$R_i = (1-e^{-10\\Delta_i})^{0.55} \\times R_{max}$$\n\nNow, we can calculate the reactive moment contributions from each bolt:\n\n$$M_i = R_i \\times d_i$$\n\n$$M_i = R_{max} (1-e^{-10\\Delta_i})^{0.55} \\times d_i$$\n\n$$\\sum M_i = R_{max} \\times \\sum (1-e^{-10\\Delta_i})^{0.55} d_i$$\n\nThe applied moment, and the reactive moment are in equilibrium. Rearrange for P:\n\n$$ M_{applied} = M_{resisting}$$\n\n$$ M_p = \\sum M_i$$\n\n$$ P \\times r_o = R_{max} \\times \\sum (1-e^{-10\\Delta_i})^{0.55} d_i$$\n\n$$ P = R_{max} \\times \\frac{\\sum (1-e^{-10\\Delta_i})^{0.55} d_i}{r_o}$$\n\nLet the second term be the ICR coefficient C. **You can think of C as a constant that converts an applied load (P) to the maximum bolt demand**\n\n$$ P = R_{max} \\times C$$\n\n$$ C = \\frac{\\sum (1-e^{-10\\Delta_i})^{0.55} d_i}{r_o}$$\n\nSet $R_{max}$ equal to the bolt capacity and back-calculate the connection capacity:\n\n$$ R_{max} = R_{capacity}$$\n\n$$ P_{capacity} = R_{capacity} \\times C$$\n\nNotes:\n\n1. Despite a nonlinear bolt force-deformation, the relationship between max bolt force ($R_{max}$) and applied force ($P$) is linear. In other words, if applied force doubles, so does maximum bolt force, and vice versa. Embedded in this is the **assumption that eccentricity (e = Mz / P) will remain constant**      \n2. If we substitute 0.34 into the exponential function above, we get 0.9815 as there's a horizontal asymptote and we will never reach 1.0 exactly. We can make a simple adjustment to our \"C\" equation if we desire. Note that AISC does NOT make this adjustment as it is more conservative to set max bolt-force as $0.9815R_{max}$, effectively capping our utilization ratio to 98%.\n\n$$ (1 - e^{-10(0.34)} )^{0.55} = 0.9815 $$\n\n$$ C = \\frac{\\sum (1-e^{-10\\Delta_i})^{0.55} d_i}{0.9815 r_o}$$\n\n\n## Theoretical Background - Brandt's Method for Locating ICR\n\nThe derivation above assumes we know where ICR is. But we don't, and it is not a trivial task to find it. The [original Crawford and Kulak paper (1971)](https://ascelibrary.org/doi/10.1061/JSDEAG.0002844) is somewhat misleading. The search space for ICR is very rarely a one-dimensional line except in the very specific situation where load angle is 0 degrees. In other words, you cannot draw an orthogonal line from P to CoG, extend that line, and expect to find ICR somewhere along it. This is explained in detail by [Muir and Thornton (2004)](https://www.cives.com/cives-engineering-corporation-publications); the search space for ICR is almost always two-dimensional.\n\nLuckily for us, there exists an iterative method that converges on ICR very quickly. [Brandt's method (1982)](https://www.aisc.org/Rapid-Determination-of-Ultimate-Strength-of-Eccentrically-Loaded-Bolt-Groups) is fast and efficient, and it is what AISC uses to construct their design tables. We will implement Brandt's method here. The two key insights presented by Brandt is summarized below:\n\n**Insight #1: Elastic method also has a center of rotation and can be readily calculated**\n\nLet $$(x_{cg}, y_{cg})$$ be the coordinate of bolt group centroid, the coordinate for the elastic center of rotation (ECR) is $$(x_{cg} +a_x,  y_{cg}+a_y)$$:\n\n$$a_x = \\frac{P_y}{n} \\frac{J}{M_z}$$\n\n$$a_y = \\frac{P_x}{n} \\frac{J}{M_z}$$\n\nWhere:\n\n* $$P_x$$ = x component of applied load\n* $$P_y$$= y component of applied load\n* $$n$$ = number of bolts in bolt group\n* $$J$$ = polar moment of inertia ($$I_z$$)\n* $$M_z$$ = applied torsion\n\n**Insight #2: Elastic center of rotation (ECR) can be used as the initial guess of (ICR), subsequent improvements can be achieved as follows:**\n\nLet the initial guess be the ECR:\n\n$$x_0 = x_{cg} +a_x$$\n\n$$y_0 = y_{cg} + a_y$$\n\nAt this assumed ICR location, calculate force equilibrium. Note how moment equilibrium is enforced when we determine $$R_{max}$$ at each step.\n\n$$\\sum F_x = f_{xx} = P_x - \\sum R_{x}$$\n\n$$\\sum F_y = f_{yy} = P_y - \\sum R_{y}$$\n\nThe force summations will not be zero unless we are at the ICR, use the residual to determine successive guesses:\n\n$$x_{i+1} = x_i - \\frac{f_{yy}J}{nM_z}$$\n\n$$y_{i+1} = y_i + \\frac{f_{xx}J}{nM_z}$$\n\nRepeat until the desired tolerance is achieved:\n\n$$\\mbox{residual} = \\sqrt{(f_{xx})^2 + (f_{yy})^2} < tol$$\n\n\n\nHere is the step by step procedure:\n\n\n* **Step 1**: From applied force $(P_x, P_y, M_z)$, calculate load vector orientation ($\\theta$). Note [atan2](https://en.wikipedia.org/wiki/Atan2) is a specialized arctan function that returns within the range between -180 to 180 degrees, rather than -90 to 90 degrees. This is to obtain a correct and unambiguous value for the angle theta.\n\n$$P = \\sqrt{P_x^2 + P_y^2}$$\n\n$$\\theta = atan2(\\frac{P_y}{P_x})$$\n\n* **Step 2**: Now calculate eccentricity and its x and y components. $e_x$ and $e_y$ is used to locate the point of applied load (let's call this point P). The location of P is actually ambiguous and can be anywhere along $L(x) = P_ye_x - P_xe_y$. One method is to place P such that the line P-CoG is perpendicular to L(x). Another alternative is to assume $e_y=0$ which is what AISC assumes and what we will do by default. Note both methods lead to the same result as long as P is oriented along L(x). You can change which assumption is used by ezbolt by changing the `ecc_method` argument in `ezbolt.BoltGroup.solve()`\n\n$$e_y = 0$$\n\n$$e_x = M_z/P_y$$\n\n$$e = \\sqrt{e_x^2 + e_y^2}$$\n\n* **Step 3**: Obtain an initial guess of ICR location per Brandt's method, then calculate distance of line P-ICR ($r_o$)\n\n$$a_x = P_y \\times \\frac{I_z}{M_z N_{bolt}}$$\n\n$$a_y = P_x \\times \\frac{I_z}{M_z N_{bolt}}$$\n\n$$x_{ICR} = x_{cg} - a_x$$\n\n$$y_{ICR} = y_{cg} + a_y$$\n\n$$r_{ox} = e_x + a_x$$\n\n$$r_{oy} = e_y - a_y$$\n\n$$r_{o} = \\sqrt{r_{ox}^2 + r_{oy}^2}$$\n\n* **Step 4**: At this point, we can already compute coefficient \"C\" at our assumed ICR location.\n\n$$C = \\frac{ \\sum((1 - e^{-10 \\Delta_i})^{0.55} d_i)}{r_o}$$\n    \n* **Step 5**: Next, we need to determine the maximum bolt force ($R_{max}$) at the user-specified load magnitude. This can be done by using the moment equilibrium equation; hence why we only need to check force equilibrium at the end. Moment equilibrium is established as a matter of course by enforcing a specific value of $R_{max}$\n\n$$M_p = P_x r_{oy} - P_y  r_{ox}$$\n\n$$M_r = R_{max} \\times \\sum((1 - e^{-10 \\Delta_i})^{0.55} d_i)$$\n\n$$R_{max} = \\frac{M_p}{\\sum((1 - e^{-10 \\Delta_i})^{0.55} d_i)}$$\n\n* **Step 6**: Now that we have $R_{max}$, we can calculate the other bolt forces:\n\n$$R_i = R_{max} \\times \\sum((1 - e^{-10 \\Delta_i})^{0.55} d_i)$$\n\n* **Step 7**: Now calculate the bolt forces' x and y component to check force equilibrium:\n\n$$cos(\\theta) = d_x / d = sin(\\theta+90^o)$$\n\n$$sin(\\theta) = d_y / d = -cos(\\theta+90^o)$$\n\n$$R_x = R_i cos(\\theta+90^o) = -R_i \\frac{d_y}{d}$$\n\n$$R_y = R_i sin(\\theta+90^o) = R_i \\frac{d_x}{d}$$\n\n* **Step 8**: Calculate residual and repeat until a specific tolerance is achieved.\n\n$$\\sum F_x = f_{xx} = P_x - \\sum R_{x}$$\n\n$$\\sum F_y = f_{yy} = P_y - \\sum R_{y}$$\n\n$$\\mbox{residual} = \\sqrt{(f_{xx})^2 + (f_{yy})^2} < tol$$\n\n* **Step 9**: If equilibrium is not achieved, go back to step 4 with the following modifications\n\n$$a_x = f_{yy} \\times \\frac{I_z}{M_z N_{bolt}}$$\n\n$$a_y = f_{xx} \\times \\frac{I_z}{M_z N_{bolt}}$$\n\n$$x_{ICR,i} = x_{ICR,i-1} - a_x$$\n\n$$y_{ICR,i} = x_{ICR,i-1} + a_y$$\n\n* **Step 10**: Once ICR has been located, calculate connection capacity:\n\n$$P_{capacity} = C \\times R_{capacity}$$\n\n$$DCR = \\frac{P}{P_{capacity}} = \\frac{R_{max}}{R_{capacity}}$$\n\nEasy enough to implement. The hard part is making sure you make an even number of sign errors.\n\n\n\n\n\n\n## Assumptions and Limitations\n\n* Sign convention follows the right-hand rule. right is +X, top is +Y, counter-clockwise is positive torsion. Note that since we are only concerned with in-plane forces, only the highlighted vectors are relevant.\n\n<div align=\"center\">\n  <img src=\"https://github.com/wcfrobert/ezbolt/blob/master/doc/signconvention.png?raw=true\" alt=\"demo\" style=\"width: 75%;\" />\n</div>\n\n* Units are in (kip, in) unless otherwise noted\n* EZbolt only calculates connection capacity with respect to bolt shear. Other limit states - such as plate rupture, block shear, bearing and tearout - are not considered.\n\n\n\n## License\n\nMIT License\n\nCopyright (c) 2023 Robert Wang",
    "bugtrack_url": null,
    "license": null,
    "summary": "ezbolt - bolt force calculations in python",
    "version": "0.2.0",
    "project_urls": {
        "Bug Tracker": "https://github.com/wcfrobert/ezbolt/issues",
        "Homepage": "https://github.com/wcfrobert/ezbolt"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "edb8da43d845f1592b32640837484000a5e74f1ab1eaa37bd92bae211aa43ba6",
                "md5": "948add31cf5a806739a7fc5383bc2a1c",
                "sha256": "d5246b35b19db28a204168905d74843e4202df7851f580da7f29e42eec32ada3"
            },
            "downloads": -1,
            "filename": "ezbolt-0.2.0-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "948add31cf5a806739a7fc5383bc2a1c",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 21224,
            "upload_time": "2024-12-08T04:49:53",
            "upload_time_iso_8601": "2024-12-08T04:49:53.968210Z",
            "url": "https://files.pythonhosted.org/packages/ed/b8/da43d845f1592b32640837484000a5e74f1ab1eaa37bd92bae211aa43ba6/ezbolt-0.2.0-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "7717262535959558462b08e5dbed69a256504db3e5fc74bd948981cc1fbf732e",
                "md5": "b750f93c5ebb5ef99ca21ecec4e9f493",
                "sha256": "d562bbafbcff70a1caff1f6c3dd4d19d291b08d87f232a24abca8e76d97bb0c3"
            },
            "downloads": -1,
            "filename": "ezbolt-0.2.0.tar.gz",
            "has_sig": false,
            "md5_digest": "b750f93c5ebb5ef99ca21ecec4e9f493",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 6021924,
            "upload_time": "2024-12-08T04:49:56",
            "upload_time_iso_8601": "2024-12-08T04:49:56.081667Z",
            "url": "https://files.pythonhosted.org/packages/77/17/262535959558462b08e5dbed69a256504db3e5fc74bd948981cc1fbf732e/ezbolt-0.2.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-12-08 04:49:56",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "wcfrobert",
    "github_project": "ezbolt",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "requirements": [
        {
            "name": "contourpy",
            "specs": [
                [
                    "~=",
                    "1.2.0"
                ]
            ]
        },
        {
            "name": "cycler",
            "specs": [
                [
                    "~=",
                    "0.12.1"
                ]
            ]
        },
        {
            "name": "fonttools",
            "specs": [
                [
                    "~=",
                    "4.47.0"
                ]
            ]
        },
        {
            "name": "kiwisolver",
            "specs": [
                [
                    "~=",
                    "1.4.5"
                ]
            ]
        },
        {
            "name": "matplotlib",
            "specs": [
                [
                    "~=",
                    "3.8.2"
                ]
            ]
        },
        {
            "name": "numpy",
            "specs": [
                [
                    "~=",
                    "1.26.2"
                ]
            ]
        },
        {
            "name": "packaging",
            "specs": [
                [
                    "~=",
                    "23.2"
                ]
            ]
        },
        {
            "name": "pandas",
            "specs": [
                [
                    "~=",
                    "2.1.4"
                ]
            ]
        },
        {
            "name": "Pillow",
            "specs": [
                [
                    "~=",
                    "10.1.0"
                ]
            ]
        },
        {
            "name": "pyparsing",
            "specs": [
                [
                    "~=",
                    "3.1.1"
                ]
            ]
        },
        {
            "name": "python-dateutil",
            "specs": [
                [
                    "~=",
                    "2.8.2"
                ]
            ]
        },
        {
            "name": "pytz",
            "specs": [
                [
                    "~=",
                    "2023.3.post1"
                ]
            ]
        },
        {
            "name": "six",
            "specs": [
                [
                    "~=",
                    "1.16.0"
                ]
            ]
        },
        {
            "name": "tzdata",
            "specs": [
                [
                    "~=",
                    "2023.3"
                ]
            ]
        }
    ],
    "lcname": "ezbolt"
}
        
Elapsed time: 0.36483s