diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 050257e79b..1885c1e7b8 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -123,6 +123,10 @@ This section illustrates how our design principles translate into actual code. W #### Design by logical entity instead of product/feature +> “The most fundamental problem in computer science is problem decomposition: how to take a complex problem and divide it up into pieces that can be solved independently.” +> +> — John Ousterhout in "A Philosophy of Software Design" + This is probably our oldest and most important design principle. When designing a module or a FAST stage we look at its domain from a functional point of view: **what is the subset of resources (or modules for FAST) that fully describes one entity and allows encapsulating its full configuration?** It's a radically different approach from designing by product or feature, where boundaries are drawn around a single GCP functionality. @@ -200,6 +204,12 @@ We have several such interfaces defined for IAM, log sinks, organizational polic #### Design interfaces to support actual usage +> “When developing a module, look for opportunities to take a little bit of extra suffering upon yourself in order to reduce the suffering of your users.” +> +> “Providing choice is good, but interfaces should be designed to make the common case as simple as possible” +> +> — John Ousterhout in "A Philosophy of Software Design" + Variables should not simply map to the underlying resource attributes, but their **interfaces should be designed to match common use cases** to reduce friction and offer the highest possible degree of legibility. This translates into different practical approaches: @@ -286,6 +296,11 @@ module "project" { #### Design compact variable spaces +> "The best modules are those whose interfaces are much simpler than their implementations" +> +> — John Ousterhout in "A Philosophy of Software Design" + + Designing variable spaces is one of the most complex aspects to get right, as they are the main entry point through which users consume modules, examples and FAST stages. We always strive to **design small variable spaces by leveraging objects and implementing defaults** so that users can quickly produce highly readable code. One of many examples of this approach comes from disk support in the `compute-vm` module, where preset defaults allow quick VM management with very few lines of code, and optional variables allow progressively expanding the code when more control is needed. @@ -413,6 +428,10 @@ Each FAST stage should be designed so that it can optionally be used in isolatio #### Stage interfaces +> “The best modules are those that provide powerful functionality yet have simple interfaces.” +> +> — John Ousterhout in "A Philosophy of Software Design" + Stages are designed based on the concept of ["contracts" or interfaces](./fast/README.md#contracts-and-stages), which define what information is produced by one stage via outputs, which is then consumed by subsequent stages via variables. Interfaces are compact in size (few variables) but broad in scope (variables typically leverage maps), so that consumers can declare in variable types only the bits of information they are interested in.