-
Notifications
You must be signed in to change notification settings - Fork 6
/
make_r_docs.R
195 lines (146 loc) · 5.84 KB
/
make_r_docs.R
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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
library(devtools)
library(fs)
library(yaml)
library(tools)
library(stringr)
# function to extract the function titles from an R file
extract_function_titles <- function(r_file) {
script_content <- readLines(r_file)
titles <- list()
current_title <- NULL
for (line in script_content) {
if (str_detect(line, "^#' @title")) {
current_title <- str_replace(line, "^#' @title\\s+(.*)", "\\1")
titles <- c(titles, current_title)
}
}
return(titles)
}
# Generate R documentation with Roxygen2
document()
# define the Directory containing the R scripts
r_dir <- "R"
# Define the path to the mkdocs.yml file
yaml_file <- "mkdocs.yml"
# define the path for the r module files
module_file_path <- "docs/code/R/modules/"
# define the path for the r function files
function_file_path <- "docs/code/R/functions/"
# Check if the module path exists
if (!dir.exists(module_file_path)) {
# If it doesn't exist, create the directories
dir.create(module_file_path, recursive = TRUE)
cat("Directory created:", module_file_path, "\n")
} else {
cat("Directory already exists:", module_file_path, "\n")
}
# Check if the function path exists
if (!dir.exists(function_file_path)) {
# If it doesn't exist, create the directories
dir.create(function_file_path, recursive = TRUE)
cat("Directory created:", function_file_path, "\n")
} else {
cat("Directory already exists:", function_file_path, "\n")
}
# List all R files in the directory
r_files <- list.files(r_dir, pattern = "\\.R$", full.names = TRUE)
# Extract function titles for each R script
file_titles_list <- lapply(r_files, extract_function_titles)
names(file_titles_list) <- basename(r_files)
# list of all .Rd files in the man/ directory
rd_file_list <- dir_ls("man", type = "file")
# convert all documented functions that are stored in the man/ directory as .Rd files
# to .md files stored in docs/R_functions. Adjust the formatting of those files for later processing
for (rd_file in rd_file_list) {
# remove the man/ prefix and the .Rd
name <- sub("^man/(.*)\\.Rd$", "\\1", rd_file)
system(paste('poetry run rd2md man', function_file_path, name))
# read in the markdown file of the function/class
function_markdown <- readLines(paste0(function_file_path, name, ".md"))
# add title line to all R6 classes (because somehow this is not done by default, contrary to functions)
if (length(function_markdown) == 0 || function_markdown[1] != paste0("# `", name, "`")) {
# Add "# class_name" as the first line
function_markdown <- c(paste0("# `", name, "`"), function_markdown)
}
# Go down one layer of navigation by adding an additional # to each title line
# this is to conform with navigation standards in mkdocs, otherwise there would be no good navigation
# in the documentation
function_markdown <- sapply(function_markdown, function(line) {
if (startsWith(line, "#")) {
return(paste0("#", line))
} else {
return(line)
}
})
# Write the modified lines back to the .md file
writeLines(function_markdown, paste0(function_file_path, name, ".md"))
}
print(".Rd processing complete")
# Read the YAML file
yaml_content <- read_yaml(yaml_file)
# Loop through all R modules. title name is the name of the R file
# add module to the navigation section of the mkdocs.yml file
# create markdown files whith snippets linking to all functions/classes belonging to this module
for (title_name in names(file_titles_list)) {
# check if there is at least one documentable function in the file, if not, skip.
if(length(file_titles_list[[title_name]]) == 0) {
next
}
mod_title_name <- substr(title_name, 1, nchar(title_name) - 2) # remove .R suffix
# generate markdown file for R module which includes all functions/classes in a file
markdown_file_content <- ""
for (name in file_titles_list[[title_name]]) {
# add snippet for each function/class to the module file
markdown_file_content <- c(markdown_file_content,
paste0('--8<-- "', sub("^[^/]+/", "", function_file_path), name,'.md"') #
)
}
# write the markdown file
writeLines(markdown_file_content, paste0(module_file_path, mod_title_name, ".md"))
code_index <- NA
r_doc_index <- NA
# Find the index of the 'Code' section in the mkdocs.yml file
for (i in seq_along(yaml_content$nav)) {
if (is.list(yaml_content$nav[[i]]) && "Code" %in% names(yaml_content$nav[[i]])) {
code_index = i
}
}
# if there is no Äcode section, create it
if (is.na(code_index)) {
yaml_content$nav <- append(yaml_content$nav, list(list(Code = NULL)))
code_index <- length(yaml_content$nav)
}
item <- yaml_content$nav[[code_index]]
# finde the index of the 'R' section in the code section
for (j in seq_along(item$Code)) {
if (is.list(item$Code[[j]]) && "R" %in% names(item$Code[[j]])) {
r_doc_index <- j
}
}
# if there is no 'R section', create it
if (is.na(r_doc_index)) {
item$Code <- append(item$Code , list(list(R=NULL)))
r_doc_index <- length(item$Code)
}
subitem <- item$Code[[r_doc_index]]
temp_list <- list()
temp_list[[mod_title_name]] <- paste0(sub("^[^/]+/", "", module_file_path), mod_title_name, '.md')
# counter to check if a page with modified_name already exists
dir_exists_counter <- FALSE
# loop through directory and override file with modified_name with the new version
for (k in seq_along(subitem$R)) {
if(names(subitem$R[[k]]) == mod_title_name) {
# if the page exists, override the entry with the new version
subitem$R[[k]] <- temp_list
dir_exists_counter <- TRUE
}
}
# if there was no file to override add a new entry with the new file
if(dir_exists_counter == FALSE) {
subitem$R[[length(subitem$R) + 1]] <- temp_list
}
yaml_content$nav[[code_index]]$Code[[r_doc_index]] <- subitem
}
# Write the modified content back to the YAML file
write_yaml(yaml_content, yaml_file)
print("made_r_docs")