/
Commented Groovy Example
Commented Groovy Example
The following example explains some of the Groovy features you can use in PFX calculation logics.
We have competitor information stored in a Product Extension called 'Competition', along with a price and validity date. There can be multiple records per competitor, each with different validity and price. We want to create a table with each competitor and its latest price (highest validity date). If two records have the same validity date, pick the higher one.
The input data is in a Product Extension called 'Competition' and the structure looks like this:
SKU | Competitor (attribute1) | Date (attribute2) | Price (attribute3) |
---|---|---|---|
00001 | Volvo | 2016-01-01 | 5698.5 |
00001 | Volvo | 2016-06-01 | 5702.0 |
00001 | Iveco | 2016-03-01 | 6041.0 |
00001 | Iveco | 2015-12-01 | 6100.2 |
00001 | MAN | 2016-05-01 | 5998.0 |
00001 | MAN | 2016-05-01 | 5713.5 |
The final result should look like this:
Competitor | Price |
---|---|
Iveco | 6041.0 |
MAN | 5998.0 |
Volvo | 5702.0 |
This is how it could be implemented in Groovy:
// get the look-back perios - number of months to look back // utilize the Elvis operator - in case the user entry is null or zero, lookback will be 12 def lookback = api.decimalUserEntry('Look-back Period [months]') ?: 12 // create the look-back date by utilizing Joda DateTime and subtracting the look-back period from the current date def lookbackDate = new DateTime().plusMonths(-lookback.toInteger()).toDate() // get all competitor records from the Product Extension whose validity is greater or equal to the look-back date // we need to convert the date to its String representation because in the PX all attributes are stored as Strings def competitors = api.productExtension('Competition', Filter.greaterOrEqual('attribute2', lookbackDate.format('yyyy-MM-dd'))) // the competitors will evaluate to true if it's a non-null and non-empty collection if (competitors) { // sort the competitors by the date (ascending) first and price (ascending) second using a custom comparator // use a Groovy Closure for that - a, b represent the two objects being compared def sortedCompetitors = competitors.sort{ a, b -> // the dates are stored as Strings in format 'yyyy-MM-dd' so it's safe to compare them lexicographically def date1 = a.attribute2 def date2 = b.attribute2 // we need to convert the prices from String to BigDecimal for it to be compared properly // the save navigation operator ?. will evaluate the entire expression to null if attribute3 is null, preventing a Null Pointer Exception def price1 = a.attribute3?.toBigDecimal() def price2 = b.attribute3?.toBigDecimal() // the Spaceship operator delegates to the compareTo method of its operands // notice how we combine it with the Elvis operator // if the comparison of the dates is zero (equal dates), which evaluates to false, it will compare the prices return date1 <=> date2 ?: price1?.toBigDecimal() <=> price2?.toBigDecimal() } // here we convert the list of competitors sorted by date and price to a map where the competitor name is the key and the competitor record is the value // the result will be a map where each competitor will have the latest price // that's because we sorted the list earlier in ascending order, so each record with later date or higher price overwrites the previous one def latestPricePerCompetitor = sortedCompetitors.collectEntries { // the left side will evaluate to the String representation thanks to the parentheses // if you don't explicitly name the iterator in the Closure, you can reference it by 'it' [ (it.attribute1): it ] } // we now create a matrix which will have three columns def columns = [ 'Competitor', 'Date', 'Price' ] // since the method api.newMatrix doesn't accept a List but an array of Strings, we use the Spread Operator to unwrap each element of the List and pass it to the method // (of course we could type in the column names directly but we needed to demonstrate the Spread operator) def matrix = api.newMatrix(*columns) // again, we use a Closure to iterate through each element of the map // but this time we rename the default 'it' to 'c' to demonstrate this feature latestPricePerCompetitor.each{ name, record -> // we add a row to the matrix for each entry in the map matrix.addRow([ 'Competitor' : name, 'Date' : record.attribute2, 'Price' : record.attribute3 ]) } // return the matrix return matrix } return null
, multiple selections available,
Related content
Groovy Cheatsheet
Groovy Cheatsheet
Read with this
General Queries (Quick Reference)
General Queries (Quick Reference)
More like this
Input Builders
Input Builders
Read with this
General Data Queries
General Data Queries
Read with this
3. ValidFrom
3. ValidFrom
More like this
Create New Calculation Engine Implementation
Create New Calculation Engine Implementation
More like this
Found an issue in documentation? Write to us.