Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Split Machinery and Tabulation - Dev Guide #614

Merged
merged 51 commits into from
Nov 1, 2023
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
d915614
init
Melkiades Apr 21, 2023
fd1df81
init 2
Melkiades Apr 21, 2023
ce4272d
first example
Melkiades Apr 21, 2023
c6f2b00
more code
Melkiades Apr 21, 2023
7d2246a
more
Melkiades Apr 21, 2023
14f9cb6
modifications after discussions
Melkiades Apr 21, 2023
7377e2e
corrections from Gabe and rephrasing
Melkiades Apr 27, 2023
c504274
into inner
Melkiades Apr 27, 2023
6b41e3c
temporary update
Melkiades May 3, 2023
a1c1d07
going further
Melkiades May 3, 2023
d3ae1bd
some fixes
Melkiades May 3, 2023
ae4c99d
Merge branch 'main' into 568_dev_guide_split@main
Melkiades May 5, 2023
32f32b5
update
Melkiades May 11, 2023
052a4d2
Merge branch 'main' into 568_dev_guide_split@main
Melkiades May 17, 2023
bcf1615
small fix
Melkiades May 17, 2023
6bd3bd5
Merge branch 'main' into 568_dev_guide_split@main
Melkiades May 24, 2023
8db2418
recent update
Melkiades May 24, 2023
c9d4a62
few mods
Melkiades Jun 9, 2023
5061819
Merge branch 'main' into 568_dev_guide_split@main
Melkiades Jun 12, 2023
c8199a9
Merge branch '568_dev_guide_split@main' of github.com:Roche/rtables i…
Melkiades Jun 12, 2023
024fd30
most recent update
Melkiades Jun 14, 2023
091c37e
Merge branch 'main' into 568_dev_guide_split@main
Melkiades Jul 19, 2023
51de558
finishing up
Melkiades Jul 19, 2023
9d8890a
comment updates
Melkiades Jul 19, 2023
f356077
more details
Melkiades Jul 20, 2023
98d2985
constructing separate vignette for debugging
Melkiades Jul 26, 2023
570c86d
adding to docs
Melkiades Jul 26, 2023
2c75dc8
init
Melkiades Jul 26, 2023
fa0d41e
init 2
Melkiades Jul 26, 2023
aac6f61
up
Melkiades Jul 26, 2023
4cabc5e
Merge branch 'main' into 568_dev_guide_split@main
Melkiades Jul 26, 2023
c9cba0e
update
Melkiades Aug 2, 2023
b4e72ce
few things
Melkiades Aug 2, 2023
d6d09fb
update
Melkiades Aug 10, 2023
6dd82f9
Merge branch 'main' into 568_dev_guide_split@main
Melkiades Aug 17, 2023
698f928
up
Melkiades Aug 17, 2023
7d7b26c
completed dev guide for tabulation
Melkiades Aug 18, 2023
14275b3
Merge branch '568_dev_guide_split@main' of github.com:Roche/rtables i…
Melkiades Aug 18, 2023
201d902
update
Melkiades Aug 21, 2023
a7f0cfe
Move files to inst/dev-guide
edelarua Oct 13, 2023
3bf0329
Update output format, render location
edelarua Oct 13, 2023
3b496a7
Merge branch 'main' into 568_dev_guide_split@main
Melkiades Oct 24, 2023
5031687
kind of completing splits
Melkiades Oct 26, 2023
32a73e2
fix debugging
Melkiades Oct 26, 2023
3ce7123
Remove from pkgdown config - fix checks
edelarua Oct 26, 2023
83546e9
Merge branch 'main' into 568_dev_guide_split@main
Melkiades Oct 31, 2023
c66501e
Split machinery - grammar fixes, adding links, applying styler
edelarua Oct 31, 2023
9148d21
Change default spaces in rtables project
edelarua Oct 31, 2023
d380128
Tabulation - grammar fixes, rewording, adding links, applying styler
edelarua Oct 31, 2023
e601ce8
Change word
edelarua Oct 31, 2023
b364d87
Debugging - grammar fixes
edelarua Oct 31, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 0 additions & 7 deletions _pkgdown.yml
Original file line number Diff line number Diff line change
Expand Up @@ -46,13 +46,6 @@ articles:
- manual_table_construction
- tabulation_dplyr

- title: For Developers
desc: Vignettes aimed at package developers
contents:
- dg_debug_rtables
- dg_split_machinery
- dg_tabulation

reference:
- title: Argument Conventions
desc: The following dummy functions are unexported and used to document argument conventions in the framework.
Expand Down
31 changes: 19 additions & 12 deletions inst/dev-guide/dg_debug_rtables.Rmd
Original file line number Diff line number Diff line change
Expand Up @@ -17,21 +17,25 @@ knitr::opts_chunk$set(echo = TRUE)

## Debugging

This is a short and not-comprehensive guide to debugging `rtables`. It is to be considered, though, of general validity and of personal use.
This is a short and non-comprehensive guide to debugging `rtables`. Regardless, it is to be considered valid for personal use at your discretion.

#### Coding in Practice
-> it is easy to read and find problems
-> it is not clever, because it is impossible to debug

* It is easy to read and find problems
* It is not clever if it is impossible to debug

#### Some Definitions

* __Coding Error__ - Code does not do what you intended -> Bug in the punch card
* __Unexpected Input__ - Defensive programming FAIL FAST FAIL LOUD (FFFL) -> useful and not too time consuming
* __Bug in Dependency__ -> never use dependencies if you can!

#### Considerations About FFFL

Errors should be as close as possible to the source. For example, bad inputs should be found very early. The worst possible example is a software that is silently giving incorrect results. Common things that we can catch early are missing values, column `length == 0`, or `length > 1`.

#### General Suggestions

* Robust code base does not attempt doing possibly problematic operations.
* Read Error Messages
* `debugcall` you can add the signature (formals)
Expand All @@ -44,6 +48,7 @@ Errors should be as close as possible to the source. For example, bad inputs sho
as you did recover.

#### `warn` Global Option

- `<0` ignored
- `0` top level function call
- `1` immediately as they occur
Expand All @@ -52,33 +57,35 @@ as you did recover.
`<<-` for `recover` or `debugger` gives it to the global environment

#### lo-fi debugging

* PRINT / CAT is always a low level debugging that can be used. It is helpful for server jobs where maybe only terminal or console output is available and no `browser()` can be used. For example, you can print the position or state of a function at a certain point untill you find the break point.
* comment blocks -> does not work with pipes (you can use `identity()` it is a step that does nothing but does not break the pipes)
* `browser()` bombing

#### Regression Tests

Almost every bug should become a regression test.

#### Debugging with Pipes

* Pipes are better to write code but horrible to debug
* T in pipe `%T>%` does print it midway
* `debug_pipe()` -> it is like the T pipe going into browser()

#### Shiny Debugging
More difficult due to reactivity.

More difficult due to reactivity.

#### General Suggestion
DO NOT BE CLEVER WITH CODE - ONLY IF YOU HAVE TO, CLEVER IS ALSO SUBJECTIVE
AND IT WILL CHANGE WITH TIME

DO NOT BE CLEVER WITH CODE - ONLY IF YOU HAVE TO, CLEVER IS ALSO SUBJECTIVE AND IT WILL CHANGE WITH TIME.

## Debugging in `rtables`

We invite the smart developer to use the provided examples as a way to get an "interactive" and dynamic vision of the internal algorithms, as they are routinely executed when constructing tables with `rtables`. This is achieved by using `browser()` and `debugonce()` on internal and exported functions (`rtables:::` or `rtables::`), as we will see in a moment. We invite also the continuous and autonomous exploration of the multiple `S3` and `S4` objects that constitute the complexity and power of `rtables`. To do so we will use the following useful functions:
We invite the smart developer to use the provided examples as a way to get an "interactive" and dynamic view of the internal algorithms as they are routinely executed when constructing tables with `rtables`. This is achieved by using `browser()` and `debugonce()` on internal and exported functions (`rtables:::` or `rtables::`), as we will see in a moment. We invite you to continuously and autonomously explore the multiple `S3` and `S4` objects that constitute the complexity and power of `rtables`. To do so, we will use the following functions:

* `methods(generic_function)`: This function lists the methods that are available for a generic function. Specifically for `S4` generic functions, `showMethods(generic_function)` is giving a more detailed information about each method (e.g. inheritance).
* `class(object)`: This function returns the class of an object. If the class is not one of the built-in classes in R, you can use this information to search for its documentation and examples. Indeed, `help(class)` may be informative, as it will call the documentation of the specific class. Similarly, the `?` operator will call the disambiguation page that delivers you to different `S4` methods. For `S3` methods it is necessary to postfix the class name with a dot (e.g. `?summary.lm`).
* `getClass(class)`: This describes in a compact way the type of class, the slots that it has, and the relationships that it may have with the other classes that may inherit or be inherited by it. With `getClass(object)`, instead, we can see to which values the slots of the object are assigned. It is possible to use `str(object, max.level = 2)` to see a less formal and more compact descriptions of the slots, but it may be problematic when there are one or more objects in the class slots. Hence, the maximum number of level should always be limited to 2 or 3 (`max.level = 2`). Similarly, `attributes()` can be used to retrieve some information, but we need to remember that storing important variables in this way is not encouraged. Information regarding the type of class can be retrieved with `mode()` and indirectly by `summary()` and `is.S4()`.
*`getAnywhere(function)` is very useful to get the source code of internal functions and specific generics. It works very well with `S3` methods, and will in any case display for each of the found methods, its relevant namespace. Similarly, `getMethod(S4_generic, S4_class)` can retrieve the source code of `S4` methods that are specific to a class.
* `eval(debugcall(generic_function(obj)))`: this is a very useful way to browser a `S4` method defined specifically for a defined object without having to manually insert `browser()` into the code. It is also possible to do similarly with R > 3.4.0 where `debug*()` calls can have the triggering signature (class) specified. Both of these are modern and simplified wrappers of tracing function `trace()`.
* `methods(generic_function)`: This function lists the methods that are available for a generic function. Specifically for `S4` generic functions, `showMethods(generic_function)` gives more detailed information about each method (e.g. inheritance).
* `class(object)`: This function returns the class of an object. If the class is not one of the built-in classes in R, you can use this information to search for its documentation and examples. `help(class)` may be informative as it will call the documentation of the specific class. Similarly, the `?` operator will bring up the documentation page for different `S4` methods. For `S3` methods it is necessary to postfix the class name with a dot (e.g. `?summary.lm`).
* `getClass(class)`: This describes the type of class in a compact way, the slots that it has, and the relationships that it may have with the other classes that may inherit from or be inherited by it. With `getClass(object)` we can see to which values the slots of the object are assigned. It is possible to use `str(object, max.level = 2)` to see less formal and more compact descriptions of the slots, but it may be problematic when there are one or more objects in the class slots. Hence, the maximum number of levels should always be limited to 2 or 3 (`max.level = 2`). Similarly, `attributes()` can be used to retrieve some information, but we need to remember that storing important variables in this way is not encouraged. Information regarding the type of class can be retrieved with `mode()` and indirectly by `summary()` and `is.S4()`.
*`getAnywhere(function)` is very useful to get the source code of internal functions and specific generics. It works very well with `S3` methods, and will display the relevant namespace for each of the methods found. Similarly, `getMethod(S4_generic, S4_class)` can retrieve the source code of class-specific `S4` methods.
* `eval(debugcall(generic_function(obj)))`: this is a very useful way to browse a `S4` method, specifically for a defined object, without having to manually insert `browser()` into the code. It is also possible to do similarly with R > 3.4.0 where `debug*()` calls can have the triggering signature (class) specified. Both of these are modern and simplified wrappers of the tracing function `trace()`.
Loading