from itertools import product from parameterized import param from parameterized import parameterized def _name_func(func, _, params): strs = [] for arg in params.args: if isinstance(arg, tuple): strs.append("_".join(str(a) for a in arg)) else: strs.append(str(arg)) # sanitize the test name name = "_".join(strs) return parameterized.to_safe_name(f"{func.__name__}_{name}") def nested_params(*params_set, name_func=_name_func): """Generate the cartesian product of the given list of parameters. Args: params_set (list of parameters): Parameters. When using ``parameterized.param`` class, all the parameters have to be specified with the class, only using kwargs. """ flatten = [p for params in params_set for p in params] # Parameters to be nested are given as list of plain objects if all(not isinstance(p, param) for p in flatten): args = list(product(*params_set)) return parameterized.expand(args, name_func=_name_func) # Parameters to be nested are given as list of `parameterized.param` if not all(isinstance(p, param) for p in flatten): raise TypeError("When using ``parameterized.param``, " "all the parameters have to be of the ``param`` type.") if any(p.args for p in flatten): raise ValueError( "When using ``parameterized.param``, " "all the parameters have to be provided as keyword argument.") args = [param()] for params in params_set: args = [param(**x.kwargs, **y.kwargs) for x in args for y in params] return parameterized.expand(args)