Skip to content

Commit

Permalink
Add SolidBody(..., block=True, apply=None)
Browse files Browse the repository at this point in the history
to apply a final callable in `SolidBody.assemble.vector()` or `SolidBody.assemble.matrix()`.
  • Loading branch information
adtzlr committed Dec 17, 2024
1 parent 8607b63 commit bfba653
Show file tree
Hide file tree
Showing 4 changed files with 52 additions and 7 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ All notable changes to this project will be documented in this file. The format
- Add a free-vibration modal analysis Step/Job `FreeVibration(items, boundaries)` with methods to evaluate `FreeVibration.evaluate()` and to extract `field, frequency = FreeVibration.extract(n)` its n-th result.
- Add `Boundary.plot()` to plot the points and prescribed directions of a boundary.
- Add `BoundaryDict` as a subclassed dict with methods to `plot()`, `screenshot()` and `imshow()`.
- Add a new argument to apply a callable on the assembled vector/matrix of a solid body, `SolidBody(..., apply=None)`. This may be used to sum the list of sub-blocks instead of stacking them together, `SolidBody(..., block=False, apply=None)`. This is useful for mixed formulations where both the deformation gradient and the displacement values are required.

### Changed
- The first Piola-Kirchhoff stress tensor is evaluated if `ViewSolid(stress_type=None)`.
Expand All @@ -17,6 +18,7 @@ All notable changes to this project will be documented in this file. The format
- Enhance `Boundary` with added support for multiaxial prescribed values.
- Enhance `math.linsteps(..., values=0)` with default values except for the column `axis` if `axis` is not None.
- Link all field-values to the values of the first field if no other field is given in `FieldContainer.link()`.
- Change the default arguments from `block=True` to `block=None` in `SolidBody.assemble.vector(block=None)` and `SolidBody.assemble.matrix(block=None)` and define `block` on creation, `SolidBody(..., block=True)` instead.

### Fixed
- Fix `Boundary(..., mode="and")` by ignoring any undefined axis.
Expand Down
7 changes: 3 additions & 4 deletions src/felupe/assembly/_integral.py
Original file line number Diff line number Diff line change
Expand Up @@ -273,13 +273,12 @@ def assemble(self, values=None, parallel=False, block=True, out=None):
if i != j:
K[j, i] = res[a].T

return bmat(K).tocsr()
res = bmat(K).tocsr()

if block and self.mode == 1:
return vstack(res).tocsr()
res = vstack(res).tocsr()

else:
return res
return res

def integrate(self, parallel=False, out=None):
if out is None:
Expand Down
47 changes: 44 additions & 3 deletions src/felupe/mechanics/_solidbody.py
Original file line number Diff line number Diff line change
Expand Up @@ -164,6 +164,11 @@ class SolidBody(Solid):
Array of initial internal state variables (default is None).
density : float or None, optional
The density of the solid body.
block : bool, optional
Assemble a sparse matrix from sparse sub-blocks or assemble a sparse vector by
stacking sparse matrices vertically (row wise). Default is True.
apply : callable or None, optional
Apply a callable on the assembled vectors and sparse matrices. Default is None.
Notes
-----
Expand Down Expand Up @@ -241,10 +246,14 @@ class SolidBody(Solid):
methods for the assembly of sparse vectors/matrices.
"""

def __init__(self, umat, field, statevars=None, density=None):
def __init__(
self, umat, field, statevars=None, density=None, block=True, apply=None
):
self.umat = umat
self.field = field
self.density = density
self.block = block
self.apply = apply

self.results = Results(stress=True, elasticity=True)
self.results.kinematics = self._extract(self.field)
Expand Down Expand Up @@ -279,32 +288,61 @@ def __init__(self, umat, field, statevars=None, density=None):
self._form = IntegralForm

def _vector(
self, field=None, parallel=False, items=None, args=(), kwargs=None, block=True
self,
field=None,
parallel=False,
items=None,
args=(),
kwargs=None,
block=None,
apply=None,
):
if kwargs is None:
kwargs = {}

if field is not None:
self.field = field

if block is None:
block = self.block

if apply is None:
apply = self.apply

self.results.stress = self._gradient(field, args=args, kwargs=kwargs)
self.results.force = self._form(
fun=self.results.stress[slice(items)],
v=self.field,
dV=self.field.region.dV,
).assemble(parallel=parallel, block=block)

if apply is not None:
self.results.force = apply(self.results.force)

Check warning on line 320 in src/felupe/mechanics/_solidbody.py

View check run for this annotation

Codecov / codecov/patch

src/felupe/mechanics/_solidbody.py#L320

Added line #L320 was not covered by tests

return self.results.force

def _matrix(
self, field=None, parallel=False, items=None, args=(), kwargs=None, block=True
self,
field=None,
parallel=False,
items=None,
args=(),
kwargs=None,
block=None,
apply=None,
):
if kwargs is None:
kwargs = {}

if field is not None:
self.field = field

if block is None:
block = self.block

if apply is None:
apply = self.apply

self.results.elasticity = self._hessian(field, args=args, kwargs=kwargs)

form = self._form(
Expand All @@ -322,6 +360,9 @@ def _matrix(
values=self.results.stiffness_values, block=block
)

if apply is not None:
self.results.stiffness = apply(self.results.stiffness)

return self.results.stiffness

def _extract(self, field):
Expand Down
3 changes: 3 additions & 0 deletions tests/test_mechanics.py
Original file line number Diff line number Diff line change
Expand Up @@ -357,6 +357,9 @@ def test_solidbody_axi_incompressible():
K2 = b.assemble.matrix(**kwargs)
assert np.allclose(K1.toarray(), K2.toarray())

K3 = b.assemble.matrix(**kwargs, block=False, apply=sum)
assert np.allclose(K1.toarray(), K3.toarray())

P1 = b.results.stress
P2 = b.evaluate.gradient()
P2 = b.evaluate.gradient(u)
Expand Down

0 comments on commit bfba653

Please sign in to comment.