Skip to content

Latest commit

 

History

History
41 lines (28 loc) · 2.58 KB

README.md

File metadata and controls

41 lines (28 loc) · 2.58 KB

purescript-capability-pattern

Working minimal illustration of the Capability Design Pattern in PureScript

(Somewhat) fully fleshed-out example of the design pattern illustrated in Jordan Martinez' phenomenal collection of reference material and examples

I've added some Aff to the mix and made sure it compiles and does something tho it's quite possible that i've made mistakes somewhere that make it imperfect as a starter template: PR's welcome.

How to build and run it

spago install & spago run should do it

What's in each "Layer"?

Layer 4 - Core - Innermost layer

Strong types & pure, total functions on those types

You'd hope to write as much of your code in this layer as possible but in this skeleton it's intentionally almost empty because we're concerned with the less obvious business of adapting this bit to your application, infrastructure and runtime.

Layer 3 - Domain - Effectful functions - program and capabilities

Called "business" logic in some descriptions of this pattern this layer contains code that essentially weaves together the concrete code from Layer 4 with the abstract capabilities that can be provided differently in different scenarios, such as a logging capability that maybe goes to the console in Test but goes to a Database or a socket or systemd or a logfile in development and production.

This layer defines:

  • a program that will run in some Monad (thus giving you freedom to run it in different Monads, see above)
  • all the capabilities that it will require of the Monad in which it runs

The capabilities are like "container requirements", an API to a structure in which this program is embedded

Layer 2 (API) & Layer 1 (Infrastructure) - a complete instance of one monadic container for a program

These two layers need to be co-located in one file in PureScript.

Together they define:

  • a particular Monad in which our program from Layer 3 can be run
  • a run function that runs the program in this Monad
  • the instances for the Monad
    • Functor, Apply, Applicative, Bind & Monad can all be derived trivially
    • others that a particular Monad might need can be written explicitly
    • the instances that are required by the program in Layer 3, also will have to be written explicitly

Layer 0 - Runtime

This layer is where it all comes together. A main is called by the underlying runtime and runs the program in one or another Monad.