In his excellent talk, “Architecting Flow in Elixir”, René Föhring discussed three different approaches to passing data through a process: pipelines, the with macro, and token passing (ex. Plug and Ecto.Changeset).
With regard to the with macro he said it’s “made for those complex scenarios…where the APIs offered functions do not deliver exactly what we need.” I take it to mean that the with macro is better used with function calls against APIs existing outside of a given module or set of modules, (e.g. libraries, services, etc.)
I’m curious to know what criteria the community uses for determining which approach to use and when.
To me those 3 things are actually two. I see the token approach as a means to clean up complex code with spread out dependencies/inputs into code, which can be pipelined very well.
For with vs. pipelines I have the soft rule of the pure functional core using pipelines and with being used at the outside where you not only have to deal with data, but also side-effects.
I see it mostly as a solution for “gathering” errors. So if, for example, you need to return more than 1 error (like in Ecto) then token passing is better as it makes accumulation simpler.
Context Object - A Design Pattern for Efficient Information Sharing across Multiple System Layers
which was submitted for
The 12th Pattern Languages of Programs (PLoP) 2005
conference
This pattern provides an efficient and application transparent way of sharing information between different layers in a software system.
So calling it a (processing) context (structure) instead of a token is entirely justified (and because of bounded context I call it a Phoenix context anyway).
You’re typically designing around a context type when an earlier stage is creating information which is used by a later stage while intermediate stages completely ignore that information. If your function’s first parameter is a tuple or map that contains information that is returned as is and is never inspected then that first parameter/return value may be a context type.
Having complete control over your own function signatures it is quite easy to implement them to accept and return the context type so that these functions just work with the pipe operator (|>).
However that also couples those functions to the context type - so it’s not something one does to functions that are reused in different situations as general functions don’t make allowances for accepting and passing “context information”.
So with/1 is often used when these more general/reusable functions are stitched into a flow (typically for errors, though you can flow “out-of-band” data inside tuples around functions that don’t use it - personally I think “adaptor functions” are clearer).