This topic describes how to create custom functions that take or return a complex StreamBase data type such as list or tuple. For custom Java functions, you must either:
-
Create a custom resolver annotation and class.
-
Register with a
pluginFunctions
section of an SBEngine type configuration file.
Custom Java functions can take or return a complex data type such as list or tuple. There are two ways you can make use of such functions in a StreamBase application:
-
Add a custom function resolver annotation and a resolver method to your function's Java code, as described in Java Functions: Using Custom Function Resolvers.
-
Register the function's argument and/or return types with a
<custom-function>
section of the server configuration file, as described in Java Functions: Using Configuration File Elements.
Java functions do not need to have pluginFunctions
elements in the SBEngine configuration file if the functions will only be called by means of the calljava() function. When used, pluginFunctions
elements are typically constructed with the autoArguments:"true"
and alias:
attributes for the purpose of defining an alias for the function — which avoids having to use calljava()
to invoke the function.
If a Java function takes or returns a complex data type, and you do not want to use a custom function resolver, then you must
specify a pluginFunctions
element for the function, and must not use the autoArguments:"true"
attribute. You must then specify argumentTypes
and returnType
elements to describe each argument and return type.
Use the configuration file definition method only when your list arguments or returns have a single, non-changing list element type, such as an argument of type list(string) or list(double), and so on. If your function is designed to take lists of several types (any of list(int), list(long), or list(double), for example), or if your function returns a list whose element type is not known until the function runs, then use a custom function resolver.
There are cases where it is preferable to use argumentTypes
settings instead of a custom function resolver. For example, your function might have non-StreamBase uses, and might have
many overloaded argument signatures for its several use cases. If you only want to expose a subset of those signatures for
use in your StreamBase application, you can specify exactly the arguments you want to support using explicit argumentTypes
entries.
This section discusses the following subjects:
When integrating a custom Java function with StreamBase, the server tries to automatically map between StreamBase data types and their Java equivalent. For example, StreamBase int is mapped to Java int, while a StreamBase string is mapped to a to Java String.
When it comes to complex data types, the automatic mapping cannot occur. Even though the natural representation for a list(int) or list(string) in Java is a List<Integer> or List<String>, respectively, at the JVM level, these types don't exist. Both types are reduced to an untyped List after Java type erasure. For this reason, the server cannot automatically infer the type of an argument or return for parameterized data types.
The solution is to provide a custom function resolver method associated with the custom Java function. The custom function resolver method has the following purposes:
-
To determine whether it is valid to call the custom function with arguments of a particular type.
-
To specify the type of the function's result.
Custom function resolvers are called at build time for an application that refers to custom functions with associated resolvers.
A custom function's resolver method is specified using the Java CustomFunctionResolver
annotation. For example:
@CustomFunctionResolver("myCustomFnResolver") public static List myCustomFn(int i, Tuple t) { ... } public static CompleteDataType myCustomFnResolver(CompleteDataType arg1, CompleteDataType arg2) { ... }
Both the annotation and the stub of a resolver class are generated for you when you use the New StreamBase Java Functions wizard in Studio to generate function code.
A resolver takes as many arguments as the method that refers to it. The types of all of these arguments must be CompleteDataType
. The return type of the resolver must also be CompleteDataType
.
For custom simple functions, a resolver can return either:
-
null
, to indicate that the custom function cannot be called with arguments whose types are specified as the parameters to the resolver. -
A valid
CompleteDataType
that characterizes the type of value returned by the custom function.
For custom simple functions that return simple data types (that is, the Java equivalents of the simple StreamBase types blob,
bool, double, int, long, string, and timestamp), the returned CompleteDataType
must be the appropriate corresponding simple type. For example, the return for an int must be as returned by CompleteDataType.forInt()
).
For custom simple functions that return complex data types (that is, the Java equivalents of the complex StreamBase types
tuple and list), the returned CompleteDataType
must fully describe the appropriate complex type (as returned by CompleteDataType.forTuple(Schema)
or CompleteDataType.forList(CompleteDataType)
).
For custom aggregate functions, only the accumulate()
methods can have associated resolvers. Even though the accumulate()
methods return no value, the return type of their resolvers (if present) must be a CompleteDataType
. If a resolver returns a non-null value, the value is taken to represent the return type of the custom aggregate function
as a whole, when used with arguments of the type specified by the chosen accumulate()
method.
See the Javadoc for CustomFunctionResolver
for more information.
Consider the following example:
class ExprUtil { @CustomFunctionResolver("concatenateResolver") public static List<?> concatenate(List<?> v1, List<?> v2) { List<?> res = new ArrayList<?>(); if (v1 != null) { res.addAll(v1); } if (v2 != null) { res.addAll(v2); } return res; } public static CompleteDataType concatenateResolver( CompleteDataType arg1, CompleteDataType arg2) { if (arg1.getElementType() == null) { return null; // not a list } if (arg1.equals(arg2)) { return arg1; } return null; } }
The concatenate()
function defined in this example takes two list arguments, as long as both lists have the same element type. The function
returns a list of the same element type.
We use the @CustomFunctionResolver
to annotate the concatenate()
method with a function resolver, which is the concatenateResolver()
method. At typecheck time, the resolver is called as necessary to determine whether the function's arguments are consistent
with the definition of the function. Then at runtime, the concatenate()
function is called appropriately.