|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
"""Preprocessing builder.""" |
|
|
|
from absl import logging |
|
from big_vision.pp import registry |
|
import tensorflow as tf |
|
|
|
|
|
def get_preprocess_fn(pp_pipeline, log_data=True): |
|
"""Transform an input string into the preprocessing function. |
|
|
|
The minilanguage is as follows: |
|
|
|
fn1|fn2(arg, arg2,...)|... |
|
|
|
And describes the successive application of the various `fn`s to the input, |
|
where each function can optionally have one or more arguments, which are |
|
either positional or key/value, as dictated by the `fn`. |
|
|
|
The output preprocessing function expects a dictionary as input. This |
|
dictionary should have a key "image" that corresponds to a 3D tensor |
|
(height x width x channel). |
|
|
|
Args: |
|
pp_pipeline: A string describing the pre-processing pipeline. If empty or |
|
None, no preprocessing will be executed. |
|
log_data: Whether to log the data before and after preprocessing. Can also |
|
be a string to show in the log for debugging, for example dataset name. |
|
|
|
Returns: |
|
preprocessing function. |
|
|
|
Raises: |
|
ValueError: if preprocessing function name is unknown |
|
""" |
|
|
|
names, ops = [], [] |
|
if pp_pipeline: |
|
for op_spec in pp_pipeline.split("|"): |
|
if not op_spec: continue |
|
try: |
|
ops.append(registry.Registry.lookup(f"preprocess_ops.{op_spec}")()) |
|
names.append(registry.parse_name(op_spec)[0]) |
|
except SyntaxError as err: |
|
raise ValueError(f"Syntax error on: {op_spec}") from err |
|
|
|
def _preprocess_fn(data): |
|
"""The preprocessing function that is returned.""" |
|
nonlocal log_data |
|
|
|
|
|
if log_data: |
|
logging.info("Data before pre-processing (%s):\n%s", log_data, data) |
|
for name, op in zip(names, ops): |
|
with tf.name_scope(name): |
|
data = op(data) |
|
|
|
|
|
if not isinstance(data, dict): |
|
raise ValueError("Argument `data` must be a dictionary, " |
|
"not %s" % str(type(data))) |
|
|
|
if log_data: |
|
logging.info("Data after pre-processing (%s):\n%s", log_data, data) |
|
log_data = False |
|
return data |
|
|
|
return _preprocess_fn |
|
|