This project was insipired by the YouTube video Transducers Explained | JavaScript by NWCalvank that references "Transducers" by Rich Hickey.
Composition is to Functional Programming (FP) what Inheritance is to Object Orientation. However, whilst the Gang of Four, in their book "Design Patterns: Elements of Reusable Object-Oriented Software" ISBN 0-201-63361-2, recommend Favor "object composition" over "class inheritance"', FP takes composition to a whole new level.
I have been familiar with the functional programming concept of lenses for years and made good use of it. Transducers are a more complicated concept but I was interested to see if they could be used to address a problem I had. This repository is the result of my investigation.
Comparator is a function used to compare two values in an array in order to determine how they should be arranged to sort the array. When the sort method is called on the array it calls the comparator with a pair of items (A, B) from the array to discover in which order they need to be arranged, as indicated by the return value.
- A negative value indicates to order A before B.
- Zero indicates that A and B are the same so the order should be left unchanged.
- Positive values indicate the order should be A after B.
Declarative can be considered to be an alternative style of processing to imperative. In the declarative style, the process is defined in terms of what needs to be achieved and the data is passed through the process.
Imperative is a style of processing that applies changes to data step by step. It is sometimes described as defining the process by how the changes are to be performed. It can be considered to be an alternative to the declarative style of processing.
Predicate is a function that takes one or more arguments and returns a Boolean value (true or false). They are very often used to define filter criteria.
Reducer is a function that combines two input values into one output value. It is often used with arrays where the first argument is a new array and the second argument is a value from the original array. In this case the output would be an extended new array.
Transform is a function that takes a single argument from which a new value is created and output. This is the form of function expected by the Array.map method.
The project is broken down into three folders:
Initial exploration into transducers applied to an array of numbers (simple) and another of objects, that uses an enhanced form of lens (lensFn).
Four collections of functions along with complete unit tests.
lens(propertyName): This function is used to produce a new function to locate a property in an object, supplied in a subsequent call, and return the properties value or undefined if not found.
lensFn(callback, propertyName): This is an enhancement of the basic lens function that takes as its first parameter a callback function. If the lens finds a value for the property in an object, the value and object are passed to the callback function.
sorter(criteria, ...)
This function generates a sort comparator function that is composed of multiple nested comparator functions. It is called with one or more criteria
objects that comprises the following properties:
- lens: A function to extract a value from an object for a given property. See the
lens
function in the utils section. - direction: Optional property that defined the order the sort is to be applied (default: ASCENDING). The ASCENDING and DESCENDING values are supplied by the sorted module.
- adaptor: Optional property for supplying a data converter for the value retreived by the lens (default: identity function
_ => _
).
{
lens: lens(_propertyName_),
direction?: DESCENDING|ASCENDING,
adaptor?: (_propertyValue_) => _convertedValue_
}
The resultant function is passed as the argument to an Array.sort method. The function is used to access a series of comparator functions in order to arrange the order of objects in an array. Subsequent comparators are called when the values being compared are equal. See the Bonus section for more information.
mapper(transform, ...) is a wrapper function that takes one or more transformation functions and returns a single transducer.
filter(predicate, ...) is another wrapper that takes aone or more predicate functions and returns a single transducer. Only objects that comply with all the predicates will remain in the output array.
extract(targetArray, retainnOriginal?)(predicate, ...) is a specialised form of filter function that takes an array into which objects from the source array are copied, and an optional flag (Boolean) indicating if objects are to be removed or retained in the source array. The initial function returns another function that takes multiple predicate functions.
conditional(predicate|condition)(transform, ...) is another specialised form of the filter function that takes a conditional function in the initial call, with (ideally) two or more transformers in the subsequent call. The conditional function returns either a Boolean (predicate) or number (0+). The Boolean is converted into a number (false = 0, true = 1) that is used as an index into the collection (array) of transformers.
composeTransducers(transducer, ...) is a function for composing transducers into a single function. The resultant function takes an array of objects, applies the functions wrapped in the transducers to each object to produce (return) a new array.
append(array, item, ... is a curried function that can be called with two arguments either separately (in subsequent calls) or together in a single call. The first parameter is the array to be appended with the appended array returned. The additional parameters (one or more) are added to the end of the initial array.
compose(transform, ...) is a utility to combine multiple functions into one.
logger(functionName, function?) is a debugging tool to output the progres through the array processing.
not(predicateFunction) is a wrapper used to invert the output of a predicate function.
pipe(transform, ...) is an alternative to compose in that is also combines functions but in the opposite (reading) order.
range(maximum, minimum?, step?) is an array generator function, used to produce test data.
lensDemo demonstrates the lens and lensFn function as described below.
exampleOne is a collection of prediate and transform functions used to perform the process that is the subject of the example. The functions are used by both implementations (imperative and declarative) without alteration.
imperative: An example solved using conventional Array methods but incuring multiple passes over the array.
declarative: The same example problem as for imperative but resolved using transducers.
temperatures.json: A raw data file for the imperative and declarative examples.
exampleData.json: A collection of several datesets, in JSON format, used to in the exampleSummary demonstration.
exampleSummary: An example with logging to demonstrate consolidating data from several sources (arrays and an object) using the transducer approach.
Lenses are functions that excepts an object and return the value of a property of the object, according to a predefined specification, or undefined if not found.
The lens function provided in the lenses library accept a property specification and returns a lens function. The specification can be a property name or a series of property names (strings). I can also accept numbers that will be assumed to be array subscripts. Strings is double-quotes are expected to be 'complex' property names that need to use the bracket notation rather than the dot notation.
For example, if we have an object as follows:
{
alpha: 'beta',
gamma: {
'delta': 'epsilon',
'zeta eta': 'theta',
},
iota: [
'kappa',
{
lambda: 'mu',
},
],
}
We can use the following specifications to retrieve values.
Specification | value |
---|---|
'alpha' | 'beta' |
'gamma', 'delta' | 'epsilon' |
'gamma.delta' | 'epsilon' |
'gamma["zeta eta"]' | 'theta' |
'gamma', '"zeta eta"' | 'theta' |
'iota[0]' | 'kappa' |
'iota', 0 | 'kappa' |
'iota', 1, 'lambda' | 'mu' |
The lenses library provides an enhanced version of the lens function called lensFn that works in a similar way but takes a call-back function as the first argument. If the property value is defined (i.e. not undefined) the function will be called with the property value and the original object. The return value of the function will be output from the generated lensFn function.
The above example can be found in the samplers folder.
A transducer is a varient of the reducer function but, when called, it returns another transducer. Their purpose being to allow different function types such as predicates and transforms to be combined into a single reducer function.
In the transducers library there are two helper functions that enable predicate and transform functions to be wrapped into transducers (filter
and mapper
respectively). These can them be composed into a new function using the composeTransducers
function. The new function takes a single array argument and through a single pass of the array, applies each of the transducers, where possible. A transform transducer will not be executed on an item if the preceeding predicate transducer returned false.
The transducers library also contains the following functions:
-
extract
works a bit likefilter
but uses the predicates to identify those items to be copied to an extracted array, which is the first parameter in the first call. The second parameter is an optional Boolean flag to indicate if the item is to be retained in the source array (defaulted to false). The subsequent call excpects to be provided with one or more predicate function that it will wrap into transducers. -
conditional
works a bit likemapper
but only one (at most) of the transforms, supplied in the second call, will be performed, depending on the zero-based index output from the conditional function, supplied as the only argument of the first call.
The following examples can be found in the samplers folder
In this example we will be converting an array of values (see below) into an array of objects containing temperatures in both Celsius and Fahrenheit. It is not important to know how to convert the temperatures. The purpose of this example is to demonstrate the difference between using a conventional imperative style verses the more declarative style provided through transducers. In both cases the inputData is supplied via the temperatures.json file as shown below.
const inputData = [
'0°C',
'',
'Invalid String',
0,
'-40°F',
null,
'273.15K',
false,
'100°C',
];
- Remove all invalid items, keeping all those that are truthy.
Result: [ '0°C', 'Invalid String', '-40°F', '273.15K', '100°C' ]
.
- Remove all items (strings) that do not match the expected pattern for a temperature in celsius or fahrenheit. Copy the invalid strings into a new array to report the exceptions.
Valid strings: [ '0°C', '-40°F', '100°C' ]
Invalid strings: [ 'Invalid String', '273.15K' ]
- Convert each valid (temperature) string into an object with the following properties: numeric (as string) and unit properties. Convert by first splitting the string at the degree symbol and assigning each element to the appropriate property.
[
{ numeric: '0', unit: 'C'},
{ numeric: '-40', unit: 'F'},
{ numeric: '100', unit: 'C'},
]
- For each temperature object, create a replacement object with a property for each scale (celsius and fahrenheit) and assigned a value, converted to a number (integer), in the appropriate scale.
[
{ celsius: 0, fahrenheit: 32},
{ celsius: -40, fahrenheit: -40},
{ celsius: 100, fahrenheit: 212},
]
In this example we will be using the Array.map and Array.filter methods to process the inputData. However, as we shall see, there are consequence from processing in this manner, that the declarative approach overcomes.
For each of the steps of the process (1-4) we need to travers an array. As a result of each traversal a new array will be produced. In addition, step 1 has to be run twice, onece to extract the invalid data for error reporting and again to isolate the valid data for further processing.
step | input | output |
---|---|---|
1 | inputData | validInputData |
2a | validInputData | invalidTemperatures |
2b | validInputData | validTemperatures |
3 | validTemperatures | temperatureSimpleObjects |
4 | temperatureSimpleObjects | temperatureComplexObjects |
Note, the output array from a step becomes the input for the next step, with the exception of the last step and step 2a.
This implementation wraps the same predicates and transforms as the imperative sampler but in this version we will wrap the functions in logger
functions before converting them into transducers.
As well as the filter
and mapper
transducers we will also use the extract
and conditional
transducers. The specialised transducers enables us to preserve a copy of the array items being filtered out of the original array to form a new array. They also enales us to be selective as to which transform to be applied based on a predicate.
In the imperative solution we passed through 4 arrays, or 5 when we include the filter to capture the invalid temperature strings, and the items in the output array will have been through as many functions.
Because the predictes and transforms functions are converted to transducers and combined using the composeTransducers
function, the solution only passes through the array once.
In the second example we are going to perform a business-orientated process. We will be bringing together several data sets to produce a summary of completed customer orders.
The exampleData.json file contains the following data sets:
customers is a list of customers who have placed orders of products to be printed. The details include a customer Id, name and their prefered method of dispatch.
(index) | customerId | companyName | dispatchPreference |
---|---|---|---|
0 | 'cust_0001' | 'Customer One' | '1st Class Postage' |
1 | 'cust_0002' | 'Customer Two' | '2nd Class Postage' |
2 | 'cust_0003' | 'Customer Three' | 'Collection' |
printJobs: Each print job held in the system details a job Id, customer Id and the value of the order (price in pence).
(index) | jobId | customerId | jobValue |
---|---|---|---|
0 | 'job_0001' | 'cust_0001' | 47500 |
1 | 'job_0002' | '' | 50000 |
2 | 'job_0003' | 'cust_0002' | 45000 |
3 | 'job_0004' | 'cust_0002' | 45000 |
4 | 'job_0005' | 'cust_0003' | 40000 |
5 | 'job_0006' | '' | 50000 |
6 | 'job_0007' | 'cust_0001' | 47500 |
7 | 'job_0008' | 'cust_0001' | 47500 |
8 | 'job_0009' | 'cust_0002' | 45000 |
9 | 'job_0010' | '' | 50000 |
10 | 'job_0011' | 'cust_0001' | 47500 |
11 | 'job_0012' | 'cust_0001' | 47500 |
12 | 'job_0013' | 'cust_0004' | 50000 |
completionReports: When a print job is complete a report is produced detailing the actual costs incured. Each report contains the job reference (job Id), how many items were printed (quantity) and the cost of the paper and ink used, in pence.
(index) | jobRef | quantity | paperCost | inkCost |
---|---|---|---|---|
0 | 'job_0001' | 100 | 11111 | 22222 |
1 | 'job_0003' | 100 | 11111 | 22222 |
2 | 'job_0005' | 100 | 11111 | 22222 |
3 | 'job_0007' | 100 | 11111 | 22222 |
4 | 'job_0009' | 100 | 11111 | 22222 |
5 | 'job_0011' | 100 | 11111 | 22222 |
postageCosts is an object containing a list of per unit costs (in pence) for differenc classes of postage. This includes an entry for when the customer has arrange collection (at zero cost) rather than paying for postage.
(index) | Values |
---|---|
1st Class Postage | 100 |
2nd Class Postage | 80 |
Collection | 0 |
The general process involves combining the information from all four data sets to distil a list of completed print jobs with all associated costs. The processed array can then be reduced to a summary report with one entry for each customer.
In addition, we want to be made aware of print jobs for which the customer is either not specificed or cannot be determined.
-
We us the
extract
function to pull out the print job that cannot be linked to a customer. -
Print jobs that have yet to be completed (i.e. do not have a completion report). This uses the
filter
function in conjunction with the completionReports data set. -
The completed print jobs are enhanced with details from the customers and completionReport data sets using the
mapper
function. -
We then calculate the postage cost according to the customer's preferred method of dispatch. This uses the
conditional
function with the postageCosts data set.
The output of the full process is too lengthy to be documented here so I suggest you try running the example yourself.
Print Run Summary
(index) | jobId | customerId | jobValue | companyName | dispatchPreference | paperCost | inkCost | products | postageCost |
---|---|---|---|---|---|---|---|---|---|
0 | 'job_0001' | 'cust_0001' | 47500 | 'Customer One' | '1st Class Postage' | 11111 | 22222 | 100 | 10000 |
1 | 'job_0003' | 'cust_0002' | 45000 | 'Customer Two' | '2nd Class Postage' | 11111 | 22222 | 100 | 8000 |
2 | 'job_0005' | 'cust_0003' | 40000 | 'Customer Three' | 'Collection' | 11111 | 22222 | 100 | 0 |
3 | 'job_0007' | 'cust_0001' | 47500 | 'Customer One' | '1st Class Postage' | 11111 | 22222 | 100 | 10000 |
4 | 'job_0009' | 'cust_0002' | 45000 | 'Customer Two' | '2nd Class Postage' | 11111 | 22222 | 100 | 8000 |
5 | 'job_0011' | 'cust_0001' | 47500 | 'Customer One' | '1st Class Postage' | 11111 | 22222 | 100 | 10000 |
Completed print job customer summary
(index) | customer | value | paper | ink | postage | totalCosts |
---|---|---|---|---|---|---|
Customer One | 'Customer One' | 142500 | 33333 | 66666 | 30000 | 129999 |
Customer Two | 'Customer Two' | 90000 | 22222 | 44444 | 16000 | 82666 |
Customer Three | 'Customer Three' | 40000 | 11111 | 22222 | 0 | 33333 |
Print Jobs without customers
(index) | jobId | customerId | jobValue |
---|---|---|---|
0 | 'job_0002' | '' | 50000 |
1 | 'job_0006' | '' | 50000 |
2 | 'job_0010' | '' | 50000 |
3 | 'job_0013' | 'cust_0004' | 50000 |
- Each printJob is only processes once.
- Only those operations necessary are applied to each printJob.
Another diffrentiator of the Functional Programming style from imperative styles is the use of (dependence on) recursion in place of loops. A perfect example of this is how we can resolve multiple sort criteria using JS's Array.sort (or the new Array.toSorted) method(s).
When a value of zero is returned by a comparator, if there are criteria remaining, the call is reissued (a recursive call) with the next criterion to perform a 'next-level' sort comparison. This continues until all criteria have been exhausted or a non-zero value is returned.
The direction argument is used as a multiplier to modify the result of the comparator and potentially invert the sort order. If an adaptor function is supplied, it is used to convert the retrieved value before the comparison is performed.
The textbook sort comparator looks something like:
function compareFn(a, b) {
if (a is less than b by some ordering criterion) {
return -1;
} else if (a is greater than b by the ordering criterion) {
return 1;
}
// a must be equal to b
return 0;
}
Since the introduction of the arrow function, the comparator is more often documented as follows.
const compareFn = (a, b) => a - b;
However, this does not lend itself to cascading the multiple sort comparators. So, I have prepared an alternative. It is based on the following simple comparator.
const compareFn = (direction) => (a, b) => direction * (+(a > b) + -(a < b));
The direction
parameter assumes one of the following values.
const ASCENDING = 1;
const DESCENDING = -1;
This approach employs partial application so instead of calling the sort method like [].sort(compareFn)
we call the function to supply the direction, as follows [].sort(compareFn(DESCENDING))
.
The algorithm is arguably more complicated than it needs to be, but it is simple when it is broken down as follows.
(a > b) // returns the Boolean value of true when _b_ needs to come before _a_.
(a < b) // returns the Boolean value of true when _a_ and _b_ are in sequence.
When a == b the above formulae both return the Boolean value of false. The plus and minus symbols in front of the formulae convert the Boolean true values to the numbers +1
and -1
. The plus in between the formulae acts as a numeric or operation so +1
and -1
persist when combined. The multiplication then provides a means of inverting the value according to the direction parameter.
(+1 + 0) * ASCENDING = +1
(0 + -1) * ASCENDING = -1
(+1 + 0) * DESCENDING = -1
(0 + -1) * DESCENDING = +1
When the result is 0 (i.e. a == b), this not only indicates the values are the same but further sort comparators might be required (if available).
The mechanism for importing JSON data via the JS import
statement is currently (May 2024) in transition. Until recently the syntax used the assert
keyword but this is being replaced by the with
keyword.
So,
import testData from './exampleData.json' assert { type: 'json' };
becomes
import testData from './exampleData.json' with { type: 'json' };