Source code for ufl.checks
"""Utility functions for checking properties of expressions."""
# Copyright (C) 2013-2016 Martin Sandve Alnæs
#
# This file is part of UFL (https://www.fenicsproject.org)
#
# SPDX-License-Identifier: LGPL-3.0-or-later
#
# Modified by Anders Logg, 2008-2009
from ufl.core.expr import Expr
from ufl.core.terminal import FormArgument
from ufl.corealg.traversal import traverse_unique_terminals
from ufl.geometry import GeometricQuantity
from ufl.sobolevspace import H1
[docs]def is_python_scalar(expression):
"""Return True iff expression is of a Python scalar type."""
return isinstance(expression, (int, float, complex))
[docs]def is_ufl_scalar(expression):
"""Return True iff expression is scalar-valued, but possibly containing free indices."""
return isinstance(expression, Expr) and not expression.ufl_shape
[docs]def is_true_ufl_scalar(expression):
"""Return True iff expression is scalar-valued, with no free indices."""
return isinstance(expression, Expr) and not (
expression.ufl_shape or expression.ufl_free_indices
)
[docs]def is_cellwise_constant(expr):
"""Return whether expression is constant over a single cell."""
# TODO: Implement more accurately considering e.g. derivatives?
return all(e.is_cellwise_constant() for e in traverse_unique_terminals(expr))
[docs]def is_scalar_constant_expression(expr):
"""Check if an expression is a globally constant scalar expression."""
if is_python_scalar(expr):
return True
if expr.ufl_shape:
return False
# TODO: This does not consider gradients of coefficients, so false
# negatives are possible.
for e in traverse_unique_terminals(expr):
# Return False if any single terminal is not constant
if isinstance(e, FormArgument):
# Accept only globally constant Arguments and Coefficients
if e.ufl_element().embedded_superdegree > 0 or e.ufl_element() not in H1:
return False
elif isinstance(e, GeometricQuantity):
# Reject all geometric quantities, they all vary over
# cells
return False
# All terminals passed constant check
return True