Please let me know if you use similar heuristics so we can learn from each other!
Use (more) functions:
If you use expressions in decisions and as the parameters for process components and plugin actions, you might reuse the same expression, or a part of it, multiple times. For example, you could define a small expression to determine whether a record status is âFinalizedâ. Instead of rewriting this expression, add it as a function to the record. If in the future the logic of âFinalizedâ will change, then you only have to update one place. (Don't repeat yourself - Wikipedia)
Use (more) process components
If any process logic is used in more than one place, encapsulate it in a Process Component. This way, you only have to update it once if you want to change it in the future (Don't repeat yourself - Wikipedia). Also, it makes your processes much more readable.
Limit decision-nesting for maintainability
Never nest deeper than three decision-layers within a process component. Nesting refers to building a large decision tree within a single process. Place underlying âsub-processesâ in their own process component with a clear name to improve maintainability, even if the process component is only called once.
Instead of this:
Build this:
Use sanity checks on input parameters
If a process component has parameters, check if each parameter is not NULL or whether they have some other weird value that would result in undesired outcomes. Do not expect the âhappy flowâ as your process can be called by anybody or by a parent process that acts weirdly. Throw an error as early as possible, and return a descriptive error message that makes debugging easy. E.g.: âCannot execute process âXXXâ because parameter ABC was not set.â
Cache once instead of querying twice
If a value is used in more than one place, cache it by using the âCacheValueâ action of the âProcess Variablesâ plugin. The fastest query/expression is the one that does not have to be executed. After retrieving a cached value, also perform a ânull check,â as it may require special handling in such cases.
Prevent slow expressions by splitting them
For example, if you want to retrieve a list of a sales that were created after the last time when product data was updated. You could write this in one expression like so:
{sales, this, createdat.isgreater(datetime:first({products,modifiedat,true,modifiedat.desc})) }
However, this can be slow as the sales and the products table are both quite big and the sub expression has the be evaluated for each sales. As the sub eypression is not depending on the parent sales, add the calculation of the date to an CacheValue and reference it:
- âCacheValue - DateTimeâ:
datetime:first({ products,modifiedat,true,modifiedat.desc })
- "CacheValue - Sales list ":
{sales, this, createdat.isgreater(cache_value_datetime.value)) }