Source code for openpnm.models.geometry.pore_size._funcs

import logging
import numpy as _np
from openpnm.models import misc as _misc
from openpnm.models.geometry import _geodocs


__all__ = [
    "weibull",
    "normal",
    "random",
    "generic_distribution",
    "from_neighbor_throats",
    "largest_sphere",
    "equivalent_diameter"
]


logger = logging.getLogger(__name__)


[docs] def weibull(network, shape, scale, loc, seeds='pore.seed'): return _misc.weibull(network, shape=shape, scale=scale, loc=loc, seeds=seeds)
weibull.__doc__ = _misc.weibull.__doc__
[docs] def normal(network, scale, loc, seeds='pore.seed'): return _misc.normal(network, scale=scale, loc=loc, seeds=seeds)
normal.__doc__ = _misc.normal.__doc__
[docs] def random(network, seed=None, num_range=[0, 1]): return _misc.random(network, element='pore', seed=seed, num_range=num_range)
random.__doc__ = _misc.random.__doc__
[docs] def generic_distribution(network, func, seeds='pore.seed'): return _misc.generic_distribution(network, func=func, seeds=seeds)
generic_distribution.__doc__ = _misc.generic_distribution.__doc__
[docs] def from_neighbor_throats(network, prop, mode='max'): return _misc.from_neighbor_throats(network, prop=prop, mode=mode)
from_neighbor_throats.__doc__ = _misc.from_neighbor_throats.__doc__
[docs] @_geodocs def largest_sphere( network, fixed_diameter='pore.fixed_diameter', iters=5 ): r""" Finds the maximum diameter pore that can be placed in each location without overlapping any neighbors. Parameters ---------- %(network)s fixed_diameter : str Name of the dictionary key on ``network`` where the array containing pore diameter values is stored, if any. If not provided a starting value is assumed as half-way to the nearest neighbor. iters : integer The number of iterations to perform when searching for maximum diameter. This function iteratively grows pores until they touch their nearest neighbor, which is also growing, so this parameter limits the maximum number of iterations. The default is 10. Returns ------- diameters : ndarray A numpy ndarray containing pore diameter values Notes ----- This function iteratively expands pores by increasing their diameter to encompass half of the distance to the nearest neighbor. If the neighbor is not growing because it's already touching a different neighbor, then the given pore will never quite touch this neighbor. Increasing the value of ``iters`` will get it closer, but it's case of Zeno's paradox with each step cutting the remaining distance in half. This model looks into all pores in the network when finding the diameter. This means that when multiple Geometry objects are defined, it will consider the diameter of pores on adjacent Geometries. If no diameters have been assigned to these neighboring pores it will assume 0. If diameter value are assigned to the neighboring pores AFTER this model is run, the pores may overlap. This can be remedied by running this model again. """ P12 = network['throat.conns'] C1 = network['pore.coords'][network['throat.conns'][:, 0]] C2 = network['pore.coords'][network['throat.conns'][:, 1]] L = _np.sqrt(_np.sum((C1 - C2)**2, axis=1)) try: # Fetch any existing pore diameters on the network D = network[fixed_diameter] # Set any unassigned values (nans) to 0 D[_np.isnan(D)] = 0 except KeyError: logger.info('Pore sizes not present, calculating starting values ' + 'as half-way to the nearest neighbor') D = _np.inf*_np.ones([network.Np, ], dtype=float) _np.minimum.at(D, P12[:, 0], L) _np.minimum.at(D, P12[:, 1], L) while iters >= 0: iters -= 1 Lt = L - _np.sum(D[P12], axis=1)/2 Dadd = _np.ones_like(D)*_np.inf _np.minimum.at(Dadd, P12[:, 0], Lt) _np.minimum.at(Dadd, P12[:, 1], Lt) D += Dadd if _np.any(D < 0): logger.info('Negative pore diameters found! Neighboring pores are ' + 'larger than the pore spacing.') return D
[docs] @_geodocs def equivalent_diameter( network, pore_volume='pore.volume', pore_shape='sphere' ): r""" Calculate the diameter of a sphere or edge-length of a cube with same volume as the pore. Parameters ---------- %(network)s %(Vp)s pore_shape : str The shape of the pore body to assume when back-calculating from volume. Options are 'sphere' (default) or 'cube'. Returns ------- diameters : ndarray A number ndarray containing pore diameter values """ from scipy.special import cbrt pore_vols = network[pore_volume] if pore_shape.startswith('sph'): value = cbrt(6*pore_vols/_np.pi) elif pore_shape.startswith('cub'): value = cbrt(pore_vols) return value