-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathreadme.Rmd
91 lines (62 loc) · 3.15 KB
/
readme.Rmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
---
title: "Static type checking in R"
output: github_document
---
```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE)
```
## R package `typeChecker`
Implements, as an experiment, a primitive static type checker following the type annotation syntax proposed in [DSC 2017 Syntax Extensions to R](https://www.r-project.org/dsc/2017/slides/syntax-extensions.html#/type-annotations) and the [types](https://github.com/jimhester/types) package.
## Installation and Usage
```{r, eval = F}
# install.packages("remotes")
remotes::install_github("kcf-jackson/typeChecker")
```
Perform type checking on a file with `type_check(filepath)`, or on the active file in RStudio using `type_check_active()`.
## Examples
#### Example 1
```{r, eval = F}
dog <- function(name = ? character) {
list(name = name)
}
dog("Archie") # correct usage
dog(123) # type error
# In the expression:
# dog(123)
# The following type error is found:
# Type mismatch. Inferred: numeric; Annotated: character
```
#### Example 2
```{r, eval = F}
dog <- function(name = ? character) {
list(name = name) ? dog
}
introduce <- function(x = ? dog) {
sprintf("Woof! My name is %s!", x$name)
}
x <- dog("Napawleon")
introduce(x) # correct usage
introduce("Pawgustus") # type error
# In the expression:
# introduce("Pawgustus")
# The following type error is found:
# Type mismatch. Inferred: character; Annotated: dog
```
## How the package works
The package type-checks R files by looking for contradiction between the annotated types and the inferred types. If no contradiction is found, then the file is type-checked. Otherwise, it will report the expression in which the error is found.
### Notes
- Type-checking in the contexts of dynamic types and static types are different.
- When there is no annotation, there cannot be any contradiction.
- An annotation is a tag, and it can be named arbitrarily.
- Constants are assigned their natural types, e.g. numeric for `10`, character for `'a'`, and logical for `TRUE`.
- All other (un-annotated) variables are assumed the ANY type, which is compatible with any type.
- Currently, the package does not support function type. In a static-type language, the type signature of a function can be identified by the types of the input and the output. In our case, the function is assigned the function type as the input and output types can vary. The important consequence is that a function that is defined but not used anywhere will not cause an error even when some expressions in the function body do not type-check.
## Integration with editors
### Visual Studio Code (VS Code)
The integration with VS Code is kindly provided by [\@andycraig](https://github.com/andycraig). The feature supports exact tracking of the line and column numbers at which the type error occurs. Here is a screenshot:
![](man/figures/vscode_demo.png)
To enable this feature, install the [R LSP Client extension](https://marketplace.visualstudio.com/items?itemName=REditorSupport.r-lsp) in VS Code and the R package `languageserver` as follows:
```
remotes::install_github("andycraig/languageserver@types")
```
The type-checking will then trigger automatically on R files.