A Python library for solving catenary equations.
For installing the latest official release on the Python Package Index (PyPI):
pip install pycatenaryFor installing a development version through pip:
git clone https://github.com/tridelat/pycatenary
cd pycatenary
pip install -e .From the root directory of pycatenary, simply run:
pytest- Catenary solutions for elastic or fully inextensible/rigid cables.
- Contact with flat floor/seabed for partly lifted lines.
- Multisegmented cables with different properties.
- Catenary lines can be defined in both 2D or 3D coordinate systems.
- Tension and position can be retrieved at fairlead/anchor and at arbitrary position along line.
- All lines, included multisegmented ones, have a single pure catenary shape.
- Gravitational acceleration is along -Z in 3D, -Y in 2D.
- If floor/seabed is enabled, it is assumed flat.
- For multisegmented lines, elongation is calculated per section, starting from the lowest segment in the catenary. Once the solution for the elongated catenary has converged, tension along the line is retrieved directly from the catenary equation using submerged weight averaged over the lifted line length.
from pycatenary import MooringLine
# define properties of cable
line1 = MooringLine(
fairlead=[-54.50, -19.84, -14.0], # fairlead position [m]
anchor=[-787.09, -286.48, -200], # anchor position [m]
L=850.0, # unstretched line length [m]
w=5844.1, # submerged weight [N/m]
EA=3.27e9, # axial stiffness [N]
floor=True, # if True, floor at anchor level
)
# compute solution for the catenary
line1.compute_solution()With matplotlib installed, the cable can be plotted in 3D:
line1.plot()Or in 2D:
line1.plot_2d()The force experienced by the anchor or fairlead at the mooring line can be retrieved as follows:
# get force at the fairlead
line1.get_fairlead_force()
# get force at the anchor
line1.get_anchor_force()Tension (strictly positive) and position can also be retrieved along the line as follows:
# get tension at 800m along the line from the anchor
line1.get_tension(800.0)
# get position at 800m along the line from the anchor
line1.get_position(800.0)
# get tension at 5m along the line from fairlead
line1.get_tension(5.0, from_fairlead=True)
# get position at 5m along the line from fairlead
line1.get_position(5.0, from_fairlead=True)Fairlead and anchor positions can be updated as follows:
# change fairlead position
line1.set_fairlead_position([-50.0, -19.84, -14.0])
# do not forget to recompute solution after updating positions
line1.compute_solution()This can be used for quasi-static analysis, retrieving forces at fairlead/anchor, applying it to an external body dynamics solver, doing a dynamics step, and updating fairlead/anchor positions of the pyCatenary line to retrieve the new forces.
For extra functionality, please refer to the documentation: https://tridelat.github.io/pycatenary
This 2D example consists of an inextensible cable hanging on its own weight between 2 points placed at the same height, with no floor:
from pycatenary import MooringLine
# define properties of cable
line2 = MooringLine(
fairlead=[17.69, 0.0],
anchor=[0.0, 0.0],
L=20.0,
w=1.962,
EA=None,
floor=False,
)
# compute solution for the catenary
line2.compute_solution()
# plot solution
line2.plot()The following 2D example showcases the different configurations possible for a mooring line.
For a partly lifted line:
from pycatenary import MooringLine
import numpy as np
# make mooring line in partly lifted line configuration
line3 = MooringLine(
fairlead=[-58.0, -14.0],
anchor=[-836.7, -200.0],
L=850.0,
w=5844.1,
EA=3.27e9,
floor=True,
)
line3.compute_solution()
line3.plot()
# move fairlead for fully lifted line position
new_position = line3.get_fairlead_position() + np.array([50.0, 0.0])
line3.set_fairlead_position(new_position)
line3.compute_solution()
line3.plot()
# move fairlead for taut line position
new_position = line3.get_fairlead_position() + np.array([100.0, 0.0])
line3.set_fairlead_position(new_position)
line3.compute_solution()
line3.plot()
# move fairlead in hanging line position
new_position = line3.get_fairlead_position() + np.array([-350.0, 0.0])
line3.set_fairlead_position(new_position)
line3.compute_solution()
line3.plot()The following 3D example consists of a mooring line with 4 segments of different properties with the floor placed at the anchor height. Segments must be defined from the anchor to the fairlead.
from pycatenary import MooringLine
import numpy as np
line4 = MooringLine(
fairlead=[0.142, 0.0, 5.486],
anchor=[7.083, 0.0, 0.0],
L=[5.672, 0.126, 4.0, 0.259],
w=np.array([0.402, 1.558, 0.00425, 1.529]) * 9.81,
EA=[2.050e3, 3.636e6, 10.873e3, 6.464e6],
floor=True
)
line4.compute_solution()
line4.plot()Note that to make the cable inextensible, the axial stiffness only needs to be defined to None as follows:
from pycatenary import MooringLine
import numpy as np
line4 = MooringLine(
fairlead=[0.142, 0.0, 5.486],
anchor=[7.083, 0.0, 0.0],
L=[5.672, 0.126, 4.0, 0.259],
w=np.array([0.402, 1.558, 0.00425, 1.529]) * 9.81,
EA=None,
floor=True
)
line4.compute_solution()To generate the documentation, run the following from the root directory:
cd docs
sphinx-build -M html buildOnce generated, open the index page from build/html/index.html.