Mercurial > repos > imgteam > image_math
changeset 0:33b2ca53a566 draft
planemo upload for repository https://github.com/BMCV/galaxy-image-analysis/tree/master/tools/image_math commit b356d76025941b691c156f8ff931cd759d35b107
author | imgteam |
---|---|
date | Sat, 09 Mar 2024 22:04:19 +0000 |
parents | |
children | f8b7770cbca5 |
files | image_math.py image_math.xml test-data/half_of_input1_plus_one.tiff test-data/input1.tiff test-data/input1_abs.tiff test-data/input1_times_2.tiff test-data/minus_input1.tiff test-data/ones.tiff |
diffstat | 8 files changed, 221 insertions(+), 0 deletions(-) [+] |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/image_math.py Sat Mar 09 22:04:19 2024 +0000 @@ -0,0 +1,90 @@ +import argparse +import ast +import operator + +import numpy as np +import skimage.io + + +supported_operators = { + ast.Add: operator.add, + ast.Sub: operator.sub, + ast.Mult: operator.mul, + ast.Div: operator.truediv, + ast.FloorDiv: operator.floordiv, + ast.Pow: operator.pow, + ast.USub: operator.neg, +} + + +supported_functions = { + 'sqrt': np.sqrt, + 'abs': abs, +} + + +def eval_ast_node(node, inputs): + """ + Evaluates a node of the syntax tree. + """ + + # Numeric constants evaluate to numeric values. + if isinstance(node, ast.Constant): + assert type(node.value) in (int, float) + return node.value + + # Variables are looked up from the inputs and resolved. + if isinstance(node, ast.Name): + assert node.id in inputs.keys() + return inputs[node.id] + + # Binary operators are evaluated based on the `supported_operators` dictionary. + if isinstance(node, ast.BinOp): + assert type(node.op) in supported_operators.keys(), node.op + op = supported_operators[type(node.op)] + return op(eval_ast_node(node.left, inputs), eval_ast_node(node.right, inputs)) + + # Unary operators are evaluated based on the `supported_operators` dictionary. + if isinstance(node, ast.UnaryOp): + assert type(node.op) in supported_operators.keys(), node.op + op = supported_operators[type(node.op)] + return op(eval_ast_node(node.operand, inputs)) + + # Function calls are evaluated based on the `supported_functions` dictionary. + if isinstance(node, ast.Call): + assert len(node.args) == 1 and len(node.keywords) == 0 + assert node.func.id in supported_functions.keys(), node.func.id + func = supported_functions[node.func.id] + return func(eval_ast_node(node.args[0], inputs)) + + # The node is unsupported and could not be evaluated. + raise TypeError(f'Unsupported node type: "{node}"') + + +def eval_expression(expr, inputs): + return eval_ast_node(ast.parse(expr, mode='eval').body, inputs) + + +if __name__ == '__main__': + + parser = argparse.ArgumentParser() + parser.add_argument('--expression', type=str, required=True) + parser.add_argument('--output', type=str, required=True) + parser.add_argument('--input', default=list(), action='append', required=True) + args = parser.parse_args() + + inputs = dict() + im_shape = None + for input in args.input: + name, filepath = input.split(':') + im = skimage.io.imread(filepath) + assert name not in inputs, 'Input name "{name}" is ambiguous.' + inputs[name] = im + if im_shape is None: + im_shape = im.shape + else: + assert im.shape == im_shape, 'Input images differ in size and/or number of channels.' + + result = eval_expression(args.expression, inputs) + + skimage.io.imsave(args.output, result)
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/image_math.xml Sat Mar 09 22:04:19 2024 +0000 @@ -0,0 +1,131 @@ +<tool id="image_math" name="Process images using arithmetic expressions" version="@TOOL_VERSION@+galaxy@VERSION_SUFFIX@" profile="23.0"> + <description>with NumPy</description> + <macros> + <token name="@TOOL_VERSION@">1.26.4</token> + <token name="@VERSION_SUFFIX@">0</token> + </macros> + <edam_operations> + <edam_operation>operation_3443</edam_operation> + </edam_operations> + <requirements> + <requirement type="package" version="1.26.4">numpy</requirement> + <requirement type="package" version="0.22.0">scikit-image</requirement> + </requirements> + <command><![CDATA[ + + ## Inputs + + python '$__tool_directory__/image_math.py' + --expression='$expression' + #for $item in $inputs: + --input='$item.name:$item.image' + #end for + + ## Outputs + + --output='./result.tiff' + + ]]> + </command> + <inputs> + <param argument="--expression" type="text" label="Expression" optional="false"> + <validator type="regex">^[a-zA-Z0-9-_\*\+ \(\)/]+$</validator> + </param> + <repeat name="inputs" title="Input images" min="1"> + <param name="image" type="data" format="png,tiff" label="Image" /> + <param name="name" type="text" label="Variable for representation of the image within the expression" optional="false"> + <validator type="regex">^[a-zA-Z_][a-zA-Z0-9_]*$</validator> + </param> + </repeat> + </inputs> + <outputs> + <data format="tiff" name="result" from_work_dir="result.tiff" /> + </outputs> + <tests> + <!-- Multiplication with a scalar --> + <test> + <param name="expression" value="input1 * 2" /> + <repeat name="inputs"> + <param name="image" value="input1.tiff" /> + <param name="name" value="input1" /> + </repeat> + <output name="result" value="input1_times_2.tiff" ftype="tiff" compare="sim_size" delta="0" /> + </test> + <!-- Unary negation operator --> + <test> + <param name="expression" value="-input1" /> + <repeat name="inputs"> + <param name="image" value="input1.tiff" /> + <param name="name" value="input1" /> + </repeat> + <output name="result" value="minus_input1.tiff" ftype="tiff" compare="sim_size" delta="0" /> + </test> + <!-- Binary addition, neutral element, addition with scalar --> + <test> + <param name="expression" value="input1 + input2 + 1" /> + <repeat name="inputs"> + <param name="image" value="input1.tiff" /> + <param name="name" value="input1" /> + </repeat> + <repeat name="inputs"> + <param name="image" value="minus_input1.tiff" /> + <param name="name" value="input2" /> + </repeat> + <output name="result" value="ones.tiff" ftype="tiff" compare="sim_size" delta="0" /> + </test> + <!-- Parentheses --> + <test> + <param name="expression" value="(input1 + input2) / 2" /> + <repeat name="inputs"> + <param name="image" value="input1.tiff" /> + <param name="name" value="input1" /> + </repeat> + <repeat name="inputs"> + <param name="image" value="ones.tiff" /> + <param name="name" value="input2" /> + </repeat> + <output name="result" value="half_of_input1_plus_one.tiff" ftype="tiff" compare="sim_size" delta="0" /> + </test> + <!-- Abs --> + <test> + <param name="expression" value="abs(input)" /> + <repeat name="inputs"> + <param name="image" value="input1.tiff" /> + <param name="name" value="input" /> + </repeat> + <output name="result" value="input1_abs.tiff" ftype="tiff" compare="sim_size" delta="0" /> + </test> + </tests> + <help> + + This tool processes images according to pixel-wise arithmetic expressions. + + The supported pixel-wise expressions are: + + - Addition, subtraction, multiplication, and division (``+``, ``-``, ``*``, ``/``) + - Integer division (e.g., ``input // 2``) + - Power (e.g., ``input ** 2``) + - Negation (e.g., ``-input``) + - Absolute values (e.g., ``abs(input)``) + - Square root (e.g., ``sqrt(input)``) + - Combinations of the above (also using parentheses) + + Examples: + + - **Negate an image.** + Expression: ``-image`` + where ``image`` is an arbitrary input image. + + - **Mean of two images.** + Expression: ``(image1 + image2) / 2`` + where ``image1`` and `image2` are two arbitrary input images. + + - **Perform division avoiding division-by-zero.** + Expression: ``image1 / (abs(image2) + 1e-8)`` + where ``image1`` and `image2` are two arbitrary input images. + + </help> + <citations> + <citation type="doi">10.1038/s41586-020-2649-2</citation> + </citations> +</tool> \ No newline at end of file