This article explains how timeouts are handled in formula a logic and in libraries which call each other and how they are handled in general.
TL;DR
...
When a formula is executed, all timeouts are reset before each of the element execution
...
The timeout specified on any library element applies to the cumulative time spent in the library element and not the time spent in each call to the library element
see example below
...
:
The executed element timeout is always honored.
Library elements timeouts are ignored. What matters is the time the logic element is executed, including library calls.
Elements timeouts are always capped by the maximum timeout configured at the cluster level.
This setting default defaults to 900s (15mn15 min).
It can be increased by Pricefx Support via the setting
formulaEngine.script.maxTimeoutInSec
The timeout of elements is set when the formula is pushed/saved to the partition
If you increased the max timeout in your cluster, don’t forget to resave your formula if needed!.
Special cases:
Since 10.0
: PA
Data Loads, PA Distributed Calculation, DMModel Calculations and Model Calculations
completely ignore the timeouts in their
Be aware that this does not apply to the libraries though!
Example when using a Library
In the example below, running libs.MyLib.ElementA.getAverageMargin()
for a large set of SKUs will time out after 2 seconds. The reason is that even though each time the getProductMargin
method is called, it takes less than 2 seconds each time, the cumulative total time spent inside that method (because of its reference in the loop) will be more than 2 seconds.
MyLib.ElementA
(timeout 300 seconds):
Code Block |
---|
Number getAverageMargin(List<String> skus){
List<Number> margins = []
for (sku in skus) {
margins.add(libs.MyLib.ElementB.getProductMargin(sku))
}
return margins.sum() / margins.size()
} |
MyLib.ElementB
(timeout 2 seconds):
Code Block |
---|
Number getProductMargin(String sku) {
def costPrice = api.find("PX30")... // get cost from some data source
def sellPrice = api.find("PLI")... // get sellprice from another data source
return sellPrice - costPrice
} |
Explanation and Technical Background
Timeouts are essentially safety catches against a bad logic (such as accidental infinite loops). JVM does not have a setting like "run this thread, but only for x seconds". So we use a trick that we inject (at compile time) some code that throws an exception if the system time has elapsed the start time + timeout. Now the different logic elements are essentially Groovy script classes. So there is no easy sharing of a script-global "start time" variable in all thinkable cases. So we resort to using a start time variable per element (normal or lib element) which is initialized at the class instantiation. So essentially a private script class member variable which is set to the system time in the constructor of the script class. The engine "resets" that start time for lib elements by re-instantiating the lib elements before executing every normal formula element. Normal elements are only instantiated once (hence the topic of "timer starts running when the element is executed first time" when you call previous elements further down).
...
logics but not in their libraries.
Since 10.2: PA Data Loads, PA Distributed Calculation, DMModel Calculations and Model Calculations completely ignore the timeouts in their logics and in the libraries they use.
Before 10.2: Elements timeout is set when the logic is pushed/saved to the partition. So if you increase the maximum timeout in your cluster, you need to resave your logic.
If an element just calls a library and the library has a higher timeout than the element, then the library element timeout is honored.