Source code for pysubsea.dnv_tools

'''
This module provides classes and functions for calculating DNV pipeline limit states and material
properties.

**Features:**

- The `DNVGeneral` class implements calculations for temperature derating, yield and tensile
  strength, and characteristic material burst strength, supporting both scalar and array-based
  inputs.
- The `DNVLimitStates` class extends `DNVGeneral` and provides burst pressure calculations for
  pipelines according to DNV standards.
- Designed for use in subsea pipeline and riser engineering, but general enough for any DNV-based
  pipeline property calculations.

All calculations are vectorized using NumPy for efficiency and flexibility.

.. raw:: html

   <hr style="height:6px; background-color:#888; border:none; margin:1.5em 0;" />

'''

import numpy as np

[docs] class DNVGeneral: # pylint: disable=too-many-instance-attributes, too-many-arguments """ Base class for DNV pipeline limit state calculations. Provides methods for temperature derating, yield and tensile strength, and characteristic material burst strength, supporting both scalar and array-based inputs. Parameters ---------- outer_diameter : float or array-like, optional The outer diameter of the pipeline. corroded_wall_thickness : float or array-like, optional The corroded wall thickness of the pipeline. material : float or array-like, optional Material types: 1 for 'CMn' or '13CR', 2 for '22Cr' or '25CR'. smys : float or array-like, optional Specified minimum yield strengths. smts : float or array-like, optional Specified minimum tensile strengths. temperature : float or array-like, optional Temperatures for calculations. material_strength_factor : float or array-like, optional Material strength factor. Notes ----- All calculations are vectorized using NumPy for efficiency and flexibility. """ def __init__( self, *, outer_diameter=0.0, corroded_wall_thickness=0.0, material=None, smys=0.0, smts=0.0, temperature=0.0, material_strength_factor=0.0 ): """ Initialize with material, strength, and geometric properties. """ self.outer_diameter = np.asarray(outer_diameter, dtype = float) self.corroded_wall_thickness = np.asarray(corroded_wall_thickness, dtype = float) self.material = np.asarray(material, dtype = float) self.smys = np.asarray(smys, dtype = float) self.smts = np.asarray(smts, dtype = float) self.temperature = np.asarray(temperature, dtype = float) self.material_strength_factor = np.asarray(material_strength_factor, dtype = float)
[docs] def temperature_derating_stress(self): """ Calculate the temperature derating stress of a material. Returns ------- temperature_derated_stress : np.ndarray The derating stress values for the given materials and temperatures. Raises ------ ValueError If a material is not supported. Examples -------- >>> materials = np.array([1, 1, 2, 2]) >>> temperatures = np.array([80.0, 110.0, 80.0, 110.0]) >>> dnv = DNVGeneral( ... material=materials, ... temperature=temperatures ... ) >>> dnv.temperature_derating_stress() array([18000000., 34000000., 70000000., 95000000.]) """ derating_array = np.empty(0) for mat, temp in zip(self.material, self.temperature): if mat == 1: derating_value = np.interp(temp, [50.0, 100.0, 200.0], [0.0, 30.0E+06, 70.0E+06]) elif mat == 2: derating_value = np.interp(temp, [20.0, 50.0, 100.0, 200.0], [0.0, 40.0E+06, 90.0E+06, 140.0E+06]) else: raise ValueError('Material not supported') derating_array = np.append(derating_array, derating_value) return derating_array
[docs] def yield_stress(self): """ Calculate the yield stress of a material. Returns ------- yield_stress : np.ndarray The yield stress values of the materials at the given temperatures and strength factor. Examples -------- >>> materials = np.array([1, 1, 2, 2]) >>> smys = np.array([450.0E+06, 450.0E+06, 550.0E+06, 550.0E+06]) >>> temperatures = np.array([80.0, 110.0, 80.0, 110.0]) >>> material_strength_factor = np.array([0.96, 0.96, 0.96, 0.96]) >>> dnv = DNVGeneral( ... material=materials, ... smys=smys, ... temperature=temperatures, ... material_strength_factor=material_strength_factor ... ) >>> dnv.yield_stress() array([4.1472e+08, 3.9936e+08, 4.6080e+08, 4.3680e+08]) """ derating_value = self.temperature_derating_stress() return (self.smys - derating_value) * self.material_strength_factor
[docs] def tensile_strength(self): """ Calculate the tensile strength of a material. Returns ------- tensile_strength : np.ndarray The tensile strength values of the materials at the given temperatures and strength factor. Examples -------- >>> materials = np.array([1, 1, 2, 2]) >>> smts = np.array([485.0E+06, 485.0E+06, 590.0E+06, 590.0E+06]) >>> temperatures = np.array([80.0, 110.0, 80.0, 110.0]) >>> material_strength_factor = np.array([0.96, 0.96, 0.96, 0.96]) >>> dnv = DNVGeneral( ... material=materials, ... smts=smts, ... temperature=temperatures, ... material_strength_factor=material_strength_factor ... ) >>> dnv.tensile_strength() array([4.4832e+08, 4.3296e+08, 4.9920e+08, 4.7520e+08]) """ derating_value = self.temperature_derating_stress() return (self.smts - derating_value) * self.material_strength_factor
[docs] def characteristic_material_burst_strength(self): """ Calculate the characteristic material burst strength. Returns ------- characteristic_burst_strength : np.ndarray The characteristic material burst strength values at the given temperatures and strength factor. Examples -------- >>> materials = np.array([1, 1, 2, 2]) >>> smys = np.array([450.0E+06, 450.0E+06, 550.0E+06, 550.0E+06]) >>> smts = np.array([600.0E+06, 600.0E+06, 700.0E+06, 700.0E+06]) >>> temperatures = np.array([80.0, 110.0, 80.0, 110.0]) >>> material_strength_factor = np.array([0.96, 0.96, 0.96, 0.96]) >>> dnv = DNVGeneral( ... material=materials, ... smys=smys, ... smts=smts, ... temperature=temperatures, ... material_strength_factor=material_strength_factor ... ) >>> dnv.characteristic_material_burst_strength() array([4.1472e+08, 3.9936e+08, 4.6080e+08, 4.3680e+08]) """ yield_stress_value = self.yield_stress() tensile_strength_value = self.tensile_strength() return np.minimum(yield_stress_value, tensile_strength_value / 1.15)
[docs] class DNVLimitStates(DNVGeneral): """ Class for DNV pipeline burst pressure limit state calculations. Extends `DNVGeneral` to provide burst pressure calculations for corroded pipelines according to DNV standards, using material, geometric, and strength properties. Parameters ---------- outer_diameter : float or array-like, optional The outer diameter of the pipeline. corroded_wall_thickness : float or array-like, optional The corroded wall thickness of the pipeline. material : float or array-like, optional Material types: 1 for 'CMn' or '13CR', 2 for '22Cr' or '25CR'. smys : float or array-like, optional Specified minimum yield strengths. smts : float or array-like, optional Specified minimum tensile strengths. temperature : float or array-like, optional Temperatures for calculations. material_strength_factor : float or array-like, optional Material strength factor. Notes ----- All parameters are passed to the parent class `DNVGeneral`. """
[docs] def burst_pressure(self): """ Calculate the burst pressure of a pipeline. Returns ------- burst_pressure : np.ndarray The burst pressure values of the pipeline. Raises ------ ValueError If a material is not supported. Examples -------- >>> outer_diameter = np.array([0.2731, 0.3239, 0.2731, 0.3239]) >>> corroded_wall_thickness = np.array([0.0097, 0.0129, 0.0097, 0.0129]) >>> materials = np.array([1, 1, 2, 2]) >>> smys = np.array([450.0E+06, 450.0E+06, 550.0E+06, 550.0E+06]) >>> smts = np.array([600.0E+06, 600.0E+06, 700.0E+06, 700.0E+06]) >>> temperatures = np.array([80.0, 110.0, 80.0, 110.0]) >>> material_strength_factor = np.array([0.96, 0.96, 0.96, 0.96]) >>> dnv = DNVLimitStates( ... outer_diameter=outer_diameter, ... corroded_wall_thickness=corroded_wall_thickness, ... material=materials, ... smys=smys, ... smts=smts, ... temperature=temperatures, ... material_strength_factor=material_strength_factor ... ) >>> dnv.burst_pressure() array([35270393.70222808, 38255444.18258572, 39189326.33580898, 41841892.07470313]) """ fcb = self.characteristic_material_burst_strength() return ( (2.0 * self.corroded_wall_thickness) / (self.outer_diameter - self.corroded_wall_thickness) * fcb * 2.0 / np.sqrt(3.0) )