mirror of
https://github.com/pachli/pachli-android.git
synced 2025-01-24 06:04:07 +01:00
3a274b0594
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.
123 lines
3.4 KiB
Markdown
123 lines
3.4 KiB
Markdown
# 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.
|
|
|
|
1. Data from the network
|
|
2. Data cached locally (either to disk or memory)
|
|
3. 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:
|
|
|
|
```kotlin
|
|
/** 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.
|
|
|
|
```mermaid
|
|
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 cause `C` to be recompiled, and vice-versa
|
|
|
|
To fix this:
|
|
|
|
1. Pick one type as being "higher" in the dependency tree than the other
|
|
2. Remove the `to...()` method from the lower type, and implement it as a companion `from()` method on the higher type
|
|
|
|
In Pachli the dependency hierarchy is (higher types on the left):
|
|
|
|
```mermaid
|
|
flowchart LR
|
|
ViewData --> ViewModel --> Cache --> Network --> Core
|
|
```
|
|
|
|
so the previous example involving a network type and a cache type would instead be written as:
|
|
|
|
```kotlin
|
|
/** 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:
|
|
|
|
```mermaid
|
|
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:
|
|
|
|
1. Functions have names that can more clearly indicate their intent
|
|
2. Functions can return objects of any subtype. If the example class `C` had multiple subtypes the correct subtype could be returned based on properties of `N`
|
|
3. 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.
|
|
4. Functions can have more specific visibility modifiers
|
|
5. Functions can be marked `inline`
|