/usr/lib/python2.7/dist-packages/woo/pre/depot.py is in python-woo 1.0+dfsg1-2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 | # encoding: utf-8
from woo.dem import *
from woo.fem import *
import woo.core, woo.dem, woo.pyderived, woo.models, woo.config
import math
from minieigen import *
class CylDepot(woo.core.Preprocessor,woo.pyderived.PyWooObject):
''''Deposition of particles inside cylindrical tube. This preprocessor was created for pre-generation of dense packing inside cylindrical chamber, which is then exported to STL serving as input to `OpenFOAM <http://openfoam.org>`__ for computing permeability (pressure loss) of the layering.
The target specimen (particle packing) dimensions are :obj:`htDiam` (cylinder height and diameter); the packing is inside cylindrical wall, with inlet at the bottom and outlet at the top; this cylinder may be longer than the specimen, which is set by :obj:`extraHt`.
The height of compact packing is not known in advance precisely (it is a function of PSD, material, layering etc); the estimate is set by :obj:`relSettle`, which is used to compute :obj:`ht0`, the height of loose packing (where particles are initially generated), which then settles down in gravity.
The packing is settled when :obj:`unbalanced energy <woo.utils.unbalancedEnergy>` (the ratio of kinetic to elastic energy) drops below :obj:`unbE`.
Once settled, particles are clipped to the required height. The actual value of settlement is computed and printed (so that it can be used iteratively as input for the next simulation).
Layering specification uses the functionality of :obj:`woo.dem.LayeredAxialBias`, which distributed fractions along some axis in particle generation space. Suppose that we are to model the following two scenarios (which correspond to :obj:`preCooked` variants ``Brisbane 1`` and ``Brisbane 2``, which have spherical particles with piecewise-linear PSD distributed in layered fractions:
.. image:: fig/depot-brisbane.*
This arrangement is achieved with the following settings:
1. ``Brisbane 1``:
PSD is defined (via :obj:`PsdSphereGenerator.psdPts <woo.dem.PsdSphereGenerator.psdPts>`) to match relative height (and thus mass) of fractions (on the right), which is 0.1777 for the coarser fraction (12.5-20mm) and 0.8222 for the finer fraction. This we set by::
psdPts=[(6.3e-3,0),(12.5e-3,.82222),(20e-3,1)]
Layering is achieved by assigning :obj:`~woo.dem.LayeredAxialBias` to :obj:`bias`. The layers are distributed along the normalized height by setting :obj:`~woo.dem.LayeredAxialBias.layerSpec` to::
[VectorX([12.5e-3,20e-3,0,.1777]),VectorX([0,12.5e-3,.1777,1]
where each ``VectorX`` contains first minimum and maximum diameter, and at least one axial height range (in normalized coordinates).
2. ``Brisbane 2``:
PSD: set :obj:`~woo.dem.PsdSphereGenerator.psdPts` to::
psdPts=[(6.3e-3,0),(12.5e-3,.4111),(20e-3,1)]
Layering (:obj:`~woo.dem.LayeredAxialBias.layerSpec`) is set as::
layerSpec=[VectorX([12.5e-3,20e-3, 0,.1777,.5888,1]),VectorX([0,12.5e-3, .177777,.58888]
where the coarse fraction is distributed uniformly over both intervals in 0-0.1777 *and* 0.5888-1.0.
Resulting heights of fractions vitally depend on :obj:`relSettlement`, so it may take some experimentation to get the result right:
.. image:: fig/depot-brisbane-3d.png
'''
_classTraits=None
_PAT=woo.pyderived.PyAttrTrait # less typing
#defaultPsd=[(.007,0),(.01,.4),(.012,.7),(.02,1)]
defaultPsd=[(5e-3,.0),(6.3e-3,.12),(8e-3,.53),(10e-3,.8),(12.5e-3,.94),(20e-3,1)]
def postLoad(self,I):
if self.preCooked and (I==None or I==id(self.preCooked)):
print 'Applying pre-cooked configuration "%s".'%self.preCooked
if self.preCooked=='Brisbane 1':
self.gen=woo.dem.PsdSphereGenerator(psdPts=[(6.3e-3,0),(12.5e-3,.82222),(20e-3,1)],discrete=False)
self.bias=woo.dem.LayeredAxialBias(axis=2,fuzz=0,layerSpec=[VectorX([12.5e-3,1,0,.177777]),VectorX([0,12.5e-3,.177777,1])])
self.relSettle=.38
elif self.preCooked=='Brisbane 2':
self.gen=woo.dem.PsdSphereGenerator(psdPts=[(6.3e-3,0),(12.5e-3,.41111),(20e-3,1)],discrete=False)
self.bias=woo.dem.LayeredAxialBias(axis=2,fuzz=0,layerSpec=[VectorX([12.5e-3,1,0,.177777,.58888,1]),VectorX([0,12.5e-3,.177777,.58888])])
self.relSettle=.37
else: raise RuntimeError('Unknown precooked configuration "%s"'%self.preCooked)
self.preCooked=''
self.ht0=self.htDiam[0]/self.relSettle
# if I==id(self.estSettle):
_attrTraits=[
_PAT(str,'preCooked','',noDump=True,noGui=False,startGroup='General',choice=['','Brisbane 1','Brisbane 2'],triggerPostLoad=True,doc='Apply pre-cooked configuration (i.e. change other parameters); this option is not saved.'),
_PAT(Vector2,'htDiam',(.45,.1),unit='m',doc='Height and diameter of the resulting cylinder; the initial cylinder has the height of :obj:`ht0`, and particles are, after stabilization, clipped to :obj:`htDiam`, the resulting height.'),
_PAT(float,'relSettle',.3,triggerPostLoad=True,doc='Estimated relative height after deposition (e.g. 0.4 means that the sample will settle around 0.4 times the original height). This value has to be guessed, as there is no exact relation to predict the amount of settling; 0.3 is a good initial guess, but it may depend on the PSD.'),
_PAT(float,'ht0',.9,guiReadonly=True,noDump=True,doc='Initial height (for loose sample), computed automatically from :obj:`relSettle` and :obj:`htDiam`.'),
_PAT(woo.dem.ParticleGenerator,'gen',woo.dem.PsdSphereGenerator(psdPts=defaultPsd,discrete=False),'Object for particle generation'),
_PAT(woo.dem.SpatialBias,'bias',woo.dem.PsdAxialBias(psdPts=defaultPsd,axis=2,fuzz=.1,discrete=True),doc='Uneven distribution of particles in space, depending on their radius. Use axis=2 for altering the distribution along the cylinder axis.'),
_PAT(woo.models.ContactModelSelector,'model',woo.models.ContactModelSelector(name='linear',damping=.4,numMat=(1,1),matDesc=['everything'],mats=[woo.dem.FrictMat(density=2e3,young=2e5,tanPhi=0)]),doc='Contact model and materials.'),
_PAT(int,'cylDiv',40,'Fineness of cylinder division'),
_PAT(float,'unbE',0.005,':obj:`Unbalanced energy <woo._utils2.unbalancedEnergy>` as criterion to consider the particles settled.'),
# STL output
_PAT(str,'stlOut','',startGroup='STL output',filename=True,doc='Output file with triangulated particles (not the boundary); if empty, nothing will be exported at the end.'),
_PAT(float,'stlTol',.2e-3,unit='m',doc='Tolerance for STL export (maximum distance between ideal shape and triangulation; passed to :obj:`_triangulated.spheroidsToStl`)'),
_PAT(Vector2,'extraHt',(.5,.5),unit='m',doc='Extra height to be added to bottom and top of the resulting packing, when the new STL-exported cylinder is created.'),
_PAT(float,'cylAxDiv',-1.,'Fineness of division of the STL cylinder; see :obj:`woo.triangulated.cylinder` ``axDiv``. The defaults create nearly-square triangulation'),
]
def __init__(self,**kw):
woo.core.Preprocessor.__init__(self)
self.wooPyInit(self.__class__,woo.core.Preprocessor,**kw)
def __call__(self):
pre=self
self.postLoad(None) # ensure consistency
mat=pre.model.mats[0]
S=woo.core.Scene(
pre=self.deepcopy(),
trackEnergy=True, # for unbalanced energy tracking
dtSafety=.9,
fields=[
DemField(
gravity=(0,0,-10),
par=woo.triangulated.cylinder(Vector3(0,0,0),Vector3(0,0,pre.ht0),radius=pre.htDiam[1]/2.,div=pre.cylDiv,capA=True,capB=False,wallCaps=True,mat=mat)
)
],
engines=DemField.minimalEngines(model=pre.model)+[
woo.dem.CylinderInlet(
node=woo.core.Node((0,0,0),Quaternion((0,1,0),-math.pi/2)),
height=pre.ht0,
radius=pre.htDiam[1]/2.,
generator=pre.gen,
spatialBias=pre.bias,
maxMass=-1,maxNum=-1,massRate=0,maxAttempts=2000,materials=pre.model.mats,glColor=float('nan'),
nDo=1 # place all particles at once, then let settle it all down
),
woo.core.PyRunner(100,'import woo.pre.depot; S.pre.checkProgress(S)'),
],
)
if 'opengl' in woo.config.features: S.any=[woo.gl.Gl1_DemField(colorBy='radius')]
return S
def checkProgress(self,S):
u=woo.utils.unbalancedEnergy(S)
print("unbalanced E: %g/%g"%(u,S.pre.unbE))
if not u<S.pre.unbE: return
r,h=S.pre.htDiam[1]/2.,S.pre.htDiam[0]
# check how much was the settlement
zz=woo.utils.contactCoordQuantiles(S.dem,[.999])
print 'Compaction done, settlement from %g (loose) to %g (dense); rel. %g, relSettle was %g.'%(S.pre.ht0,zz[0],zz[0]/S.pre.ht0,S.pre.relSettle)
# delete everything abot; run this engine just once, explicitly
woo.dem.BoxOutlet(box=((-r,-r,0),(r,r,h)))(S,S.dem)
S.stop()
# delete the triangulated cylinder
for p in S.dem.par:
if isinstance(p.shape,Facet): S.dem.par.remove(p.id)
# create a new (CFD-suitable) cylinder
# bits for marking the mesh parts
S.lab.cylBits=8,16,32
cylMasks=[DemField.defaultBoundaryMask | b for b in S.lab.cylBits]
S.dem.par.add(woo.triangulated.cylinder(Vector3(0,0,-S.pre.extraHt[0]),Vector3(0,0,S.pre.htDiam[0]+S.pre.extraHt[1]),radius=S.pre.htDiam[1]/2.,div=S.pre.cylDiv,axDiv=S.pre.cylAxDiv,capA=True,capB=True,wallCaps=False,masks=cylMasks,mat=S.pre.model.mats[0]))
if S.pre.stlOut:
n=woo.triangulated.spheroidsToSTL(S.pre.stlOut,S.dem,tol=S.pre.stlTol,solid="particles")
n+=woo.triangulated.facetsToSTL(S.pre.stlOut,S.dem,append=True,mask=S.lab.cylBits[0],solid="lateral")
n+=woo.triangulated.facetsToSTL(S.pre.stlOut,S.dem,append=True,mask=S.lab.cylBits[1],solid="bottom")
n+=woo.triangulated.facetsToSTL(S.pre.stlOut,S.dem,append=True,mask=S.lab.cylBits[2],solid="top")
print 'Exported %d facets to %s'%(n,S.pre.stlOut)
else:
print 'Not running STL export (stlOut empty)'
|