The previous code generally converted between a higher and a lower type by putting the type conversion functions on the lower type. This introduced cycles in the code dependency graph, and made it more difficult to follow the code flow. Refactor the code so that types generally have a `from(...)` static factory method that can create an instance from a lower type, and if appropriate a `to...()` method that can also create an instance of that lower type. Add `docs/code-style.md` which explains the rationale for this change in more detail so that future contributors can write code in the same style.
3.4 KiB
Code style guide
Synopsis
This document describes aspects of code style that are not enforced with linters or formatting tools but the project still tries to adhere to. Some of these are things that developers might reasonably disagree on, but the project has a specific stance.
Topics
On converting between types
Roughly speaking the code that handles data from the user's server deals with three variations of that data.
- Data from the network
- Data cached locally (either to disk or memory)
- Data displayed to the user
There must be code to convert between those representations, and it's important to make sure there isn't a loop in the dependency graph between the types.
Consider two types, N
, representing data received from the network, and C
, representing data that will be cached.
The wrong way to do it is code like:
/** In N.kt (the network data type) */
import C
data class N() {
fun toC(): C { /* return a C created from a N */ }
}
// ---
/** in C.kt (the cache data type) */
import N
data class C() {
fun toN(): N { /* return a N created from a C */ }
}
This creates a loop in their dependency graph as they import each other's types.
classDiagram
direction RL
class N{
toC() C
}
class C{
toN() N
}
C ..> N: Imports
N ..> C: Imports
This is a problem because:
- They can't be placed in separate modules
- Modifying code in
N
can causeC
to be recompiled, and vice-versa
To fix this:
- Pick one type as being "higher" in the dependency tree than the other
- Remove the
to...()
method from the lower type, and implement it as a companionfrom()
method on the higher type
In Pachli the dependency hierarchy is (higher types on the left):
flowchart LR
ViewData --> ViewModel --> Cache --> Network --> Core
so the previous example involving a network type and a cache type would instead be written as:
/** In N.kt (the network data type) */
data class N() {
// No import, no toC() method
}
// ---
/** in C.kt (the cache data type) */
import N
data class C() {
fun toN(): N { /* return a N created from a C */ }
companion object {
fun from(n: N): C {
// code from the N.toC() in the previous example
}
}
}
Now the dependency between the two types is:
classDiagram
direction RL
class N{
}
class C{
toN() N
from(n: N)$ C
}
C <.. N: Imports
The circular dependency is gone, the N
type can easily be placed in a separate module to the C
type, and changes to the C
type will not require the N
type to be recompiled.
In these examples the from
method could also have been written as a secondary constructor instead of a static factory method in the companion object. We prefer static factory methods over secondary constructors because:
- Functions have names that can more clearly indicate their intent
- Functions can return objects of any subtype. If the example class
C
had multiple subtypes the correct subtype could be returned based on properties ofN
- Functions can return null or other values to signify an error. Perhaps the network type is expected to contain a particular property, but the server has a bug and returned data without that property.
- Functions can have more specific visibility modifiers
- Functions can be marked
inline