from copy import copy
import numpy as nm
import pytest
from sfepy.base.base import ordered_iteritems
from sfepy import data_dir
import sfepy.base.testing as tst
filename_meshes = [data_dir + '/meshes/elements/%s_2.mesh' % geom
for geom in ['1_2', '2_1_2', '3_1_2',
'2_3', '2_4',
'3_4', '3_8', '3_2_4']]
not_tested_terms = ['dw_ns_dot_grad_s',
'dw_tl_he_ogden_ad']
[docs]
def make_term_args(arg_shapes, arg_kinds, arg_types, ats_mode, domain,
material_value=None, poly_space_basis=None):
from sfepy.base.base import basestr
from sfepy.discrete import FieldVariable, Material, Variables, Materials
from sfepy.discrete.fem import Field
from sfepy.solvers.ts import TimeStepper
from sfepy.mechanics.tensors import dim2sym
omega = domain.regions['Omega']
dim = domain.shape.dim
sym = dim2sym(dim)
def _parse_scalar_shape(sh):
if isinstance(sh, basestr):
if sh == 'D':
return dim
elif sh == 'D2':
return dim**2
elif sh == 'S':
return sym
elif sh == 'SD':
return sym * dim
elif sh == 'N': # General number ;)
return 1
elif sh == 'str':
return 'str'
else:
return int(sh)
else:
return sh
def _parse_tuple_shape(sh):
if isinstance(sh, basestr):
return [_parse_scalar_shape(ii.strip()) for ii in sh.split(',')]
else:
return (int(sh),)
args = {}
str_args = []
materials = []
variables = []
for ii, arg_kind in enumerate(arg_kinds):
if arg_kind != 'ts':
if ats_mode is not None:
extended_ats = arg_types[ii] + ('/%s' % ats_mode)
else:
extended_ats = arg_types[ii]
try:
sh = arg_shapes[arg_types[ii]]
except KeyError:
sh = arg_shapes[extended_ats]
if arg_kind.endswith('variable'):
shape = _parse_scalar_shape(sh[0] if isinstance(sh, tuple) else sh)
field = Field.from_args('f%d' % ii, nm.float64, shape, omega,
approx_order=1,
poly_space_basis=poly_space_basis)
if arg_kind == 'virtual_variable':
if sh[1] is not None:
istate = arg_types.index(sh[1])
else:
# Only virtual variable in arguments.
istate = -1
# -> Make fake variable.
var = FieldVariable('u-1', 'unknown', field)
var.set_constant(0.0)
variables.append(var)
var = FieldVariable('v', 'test', field,
primary_var_name='u%d' % istate)
elif arg_kind == 'state_variable':
var = FieldVariable('u%d' % ii, 'unknown', field)
var.set_constant(0.0)
elif arg_kind == 'parameter_variable':
var = FieldVariable('p%d' % ii, 'parameter', field,
primary_var_name='(set-to-None)')
var.set_constant(0.0)
variables.append(var)
str_args.append(var.name)
args[var.name] = var
elif arg_kind.endswith('material'):
if sh is None: # Switched-off opt_material.
continue
prefix = ''
if isinstance(sh, basestr):
aux = sh.split(':')
if len(aux) == 2:
prefix, sh = aux
value = material_value
if value is None:
value = 1.0
elif isinstance(value, dict):
value = material_value.get(arg_types[ii], 1.0)
shape = _parse_tuple_shape(sh)
if 'str' in shape:
values = {'.c%d' % ii : value}
elif (len(shape) > 1) or (shape[0] > 1):
if ((len(shape) == 2) and (shape[0] == shape[1])
and (value != 0.0)):
# Identity matrix.
val = nm.eye(shape[0], dtype=nm.float64)
else:
# Array.
val = nm.empty(shape, dtype=nm.float64)
val.fill(value)
values = {'%sc%d' % (prefix, ii)
: val}
elif (len(shape) == 1) and (shape[0] == 1):
# Single scalar as a special value.
values = {'.c%d' % ii : value}
else:
raise ValueError('wrong material shape! (%s)' % shape)
mat = Material('m%d' % ii, values=values)
materials.append(mat)
str_args.append(mat.name + '.' + 'c%d' % ii)
args[mat.name] = mat
elif arg_kind == 'ts':
ts = TimeStepper(0.0, 1.0, 1.0, 5)
str_args.append('ts')
args['ts'] = ts
else:
str_args.append('user%d' % ii)
args[str_args[-1]] = sh if callable(sh) else None
materials = Materials(materials)
variables = Variables(variables)
return args, str_args, materials, variables
[docs]
@pytest.fixture(scope='module')
def data():
from sfepy.base.base import Struct
from sfepy.discrete import Integral
from sfepy.discrete.fem import Mesh, FEDomain
domains = []
for filename in filename_meshes:
mesh = Mesh.from_file(filename)
domain = FEDomain('domain_%s' % mesh.name.replace(data_dir, ''),
mesh)
domain.create_region('Omega', 'all')
domain.create_region('Gamma', 'vertices of surface', 'facet')
domains.append(domain)
integral0 = Integral('i0', order=0)
integral = Integral('i', order=3)
qp_coors, qp_weights = integral.get_qp('3_8')
custom_integral = Integral('i', coors=qp_coors, weights=qp_weights,
order='custom')
return Struct(domains=domains, integral0=integral0, integral=integral,
custom_integral=custom_integral)
def _test_single_term(data, term_cls, domain, rname):
from sfepy.terms import Term
from sfepy.terms.terms import get_arg_kinds
ok = True
term_call = term_cls.name + '(%s)'
arg_shapes_list = term_cls.arg_shapes
if not isinstance(arg_shapes_list, list):
arg_shapes_list = [arg_shapes_list]
if term_cls.integration == 'custom':
integral = data.custom_integral
elif term_cls.integration_order == 0:
integral = data.integral0
else:
integral = data.integral
poly_space_basis = getattr(term_cls, 'poly_space_basis', 'lagrange')
prev_shapes = {}
for _arg_shapes in arg_shapes_list:
# Unset shapes are taken from the previous iteration.
arg_shapes = copy(prev_shapes)
arg_shapes.update(_arg_shapes)
prev_shapes = arg_shapes
tst.report('arg_shapes:', arg_shapes)
arg_types = term_cls.arg_types
if not isinstance(arg_types[0], tuple):
arg_types = (arg_types,)
for iat, ats in enumerate(arg_types):
tst.report('arg_types:', ats)
arg_kinds = get_arg_kinds(ats)
modes = getattr(term_cls, 'modes', None)
mode = modes[iat] if modes is not None else None
if 'dw_s_dot_grad_i_s' in term_cls.name:
material_value = 0.0
elif 'de_mass' in term_cls.name:
material_value = {'material_lumping' : 'row_sum'}
elif 'dw_contact_ipc' in term_cls.name:
material_value = {'material_p' : 'NONE'}
else:
material_value = 1.0
aux = make_term_args(arg_shapes, arg_kinds, ats, mode, domain,
material_value=material_value,
poly_space_basis=poly_space_basis)
args, str_args, materials, variables = aux
tst.report('args:', str_args)
name = term_call % (', '.join(str_args))
term = Term.new(name, integral, domain.regions[rname], **args)
term.setup()
call_mode = 'weak' if term.names.virtual else 'eval'
tst.report('call mode:', call_mode)
out = term.evaluate(mode=call_mode, ret_status=True)
if call_mode == 'eval':
vals, status = out
vals = nm.array(vals)
else:
vals, iels, status = out
if isinstance(vals, tuple):
# Dynamic connectivity terms.
vals = vals[0]
_ok = nm.isfinite(vals).all()
ok = ok and _ok
tst.report('values shape: %s' % (vals.shape,))
if not _ok:
tst.report('values are not finite!')
tst.report(vals)
_ok = status == 0
if not _ok:
tst.report('status is %s!' % status)
ok = ok and _ok
if term.names.virtual:
# Test differentiation w.r.t. state variables in the weak
# mode.
svars = term.get_state_variables(unknown_only=True)
for svar in svars:
vals, iels, status = term.evaluate(mode=call_mode,
diff_var=svar.name,
ret_status=True)
if isinstance(vals, tuple):
# Dynamic connectivity terms.
vals = vals[0]
_ok = status == 0
ok = ok and _ok
tst.report('diff: %s' % svar.name)
if not _ok:
tst.report('status is %s!' % status)
_ok = nm.isfinite(vals).all()
ok = ok and _ok
tst.report('values shape: %s' % (vals.shape,))
if not _ok:
tst.report('values are not finite!')
tst.report(vals)
return ok
[docs]
@pytest.mark.slow
def test_term_call_modes(data):
from sfepy.terms import term_table
ok = True
failed = []
for domain in data.domains:
tst.report('domain: %s' % domain.name)
domain_geometry = list(domain.geom_els.values())[0].name
if domain.shape.dim != domain.shape.tdim:
domain_geometry = '%d_%s' % (domain.shape.dim, domain_geometry)
for _, term_cls in ordered_iteritems(term_table):
if (domain_geometry not in term_cls.geometries) \
or ("dg" in term_cls.name) \
or (term_cls.name in not_tested_terms):
continue
rnames = []
if ('cell' in term_cls.integration or
'point' in term_cls.integration or
'custom' in term_cls.integration):
rnames.append('Omega')
if ('facet' in term_cls.integration or
'facet_extra' in term_cls.integration):
rnames.append('Gamma')
for rname in rnames:
tst.report('<-- %s.%s ...' % (term_cls.name, rname))
if rname == 'Gamma' and domain.mesh.dim == 1:
tst.report('--> 1D Gamma region: not tested!')
elif term_cls.arg_shapes:
try:
_ok = _test_single_term(data, term_cls, domain,
rname)
except ModuleNotFoundError as exc:
_ok = exc.name in {'ipctk'}
except:
_ok = False
if not _ok:
failed.append((domain.name, term_cls.name))
ok = ok and _ok
tst.report('--> ok: %s' % _ok)
else:
tst.report('--> not tested!')
tst.report('failed:', failed)
assert ok