diff --git a/.Rbuildignore b/.Rbuildignore new file mode 100644 index 0000000..62479ff --- /dev/null +++ b/.Rbuildignore @@ -0,0 +1,24 @@ +.DS_Store +.gitignore +.git +.git/* +README.md +FAQ.md +TODO.md +LICENSE.md +Makefile +.Rprofile +^\.Rproj\.user$ +^\.*\Rproj$ +^\.travis\.yml$ +^appveyor\.yml$ +^.*\.Rproj$ +^doc$ +^Meta$ +^_pkgdown\.yml$ +^docs$ +^pkgdown$ +^\.github$ +^CODE_OF_CONDUCT\.md$ +^revdep$ +data-raw diff --git a/.github/.gitignore b/.github/.gitignore new file mode 100644 index 0000000..2d19fc7 --- /dev/null +++ b/.github/.gitignore @@ -0,0 +1 @@ +*.html diff --git a/.github/workflows/R-CMD-check.yaml b/.github/workflows/R-CMD-check.yaml new file mode 100644 index 0000000..0f2fe08 --- /dev/null +++ b/.github/workflows/R-CMD-check.yaml @@ -0,0 +1,52 @@ +# Workflow derived from https://github.com/r-lib/actions/tree/v2/examples +# Need help debugging build failures? Start at https://github.com/r-lib/actions#where-to-find-help +on: + push: + branches: [main, master] + pull_request: + branches: [main, master] + +name: R-CMD-check + +permissions: read-all + +jobs: + R-CMD-check: + runs-on: ${{ matrix.config.os }} + + name: ${{ matrix.config.os }} (${{ matrix.config.r }}) + + strategy: + fail-fast: false + matrix: + config: + - {os: macos-latest, r: 'release'} + - {os: windows-latest, r: 'release'} + - {os: ubuntu-latest, r: 'devel', http-user-agent: 'release'} + - {os: ubuntu-latest, r: 'release'} + - {os: ubuntu-latest, r: 'oldrel-1'} + + env: + GITHUB_PAT: ${{ secrets.GITHUB_TOKEN }} + R_KEEP_PKG_SOURCE: yes + + steps: + - uses: actions/checkout@v4 + + - uses: r-lib/actions/setup-pandoc@v2 + + - uses: r-lib/actions/setup-r@v2 + with: + r-version: ${{ matrix.config.r }} + http-user-agent: ${{ matrix.config.http-user-agent }} + use-public-rspm: true + + - uses: r-lib/actions/setup-r-dependencies@v2 + with: + extra-packages: any::rcmdcheck + needs: check + + - uses: r-lib/actions/check-r-package@v2 + with: + upload-snapshots: true + build_args: 'c("--no-manual","--compact-vignettes=gs+qpdf")' diff --git a/.github/workflows/pkgdown.yaml b/.github/workflows/pkgdown.yaml new file mode 100644 index 0000000..c9f0165 --- /dev/null +++ b/.github/workflows/pkgdown.yaml @@ -0,0 +1,50 @@ +# Workflow derived from https://github.com/r-lib/actions/tree/v2/examples +# Need help debugging build failures? Start at https://github.com/r-lib/actions#where-to-find-help +on: + push: + branches: [main, master] + pull_request: + branches: [main, master] + release: + types: [published] + workflow_dispatch: + +name: pkgdown + +permissions: read-all + +jobs: + pkgdown: + runs-on: ubuntu-latest + # Only restrict concurrency for non-PR jobs + concurrency: + group: pkgdown-${{ github.event_name != 'pull_request' || github.run_id }} + env: + GITHUB_PAT: ${{ secrets.GITHUB_TOKEN }} + permissions: + contents: write + steps: + - uses: actions/checkout@v4 + + - uses: r-lib/actions/setup-pandoc@v2 + + - uses: r-lib/actions/setup-r@v2 + with: + use-public-rspm: true + + - uses: r-lib/actions/setup-r-dependencies@v2 + with: + extra-packages: any::pkgdown, local::. + needs: website + + - name: Build site + run: pkgdown::build_site_github_pages(new_process = FALSE, install = FALSE) + shell: Rscript {0} + + - name: Deploy to GitHub pages 🚀 + if: github.event_name != 'pull_request' + uses: JamesIves/github-pages-deploy-action@v4.5.0 + with: + clean: false + branch: gh-pages + folder: docs diff --git a/.gitignore b/.gitignore index e75435c..e16ee6e 100644 --- a/.gitignore +++ b/.gitignore @@ -4,7 +4,6 @@ # Session Data files .RData -.RDataTmp # User-specific files .Ruserdata @@ -39,11 +38,11 @@ vignettes/*.pdf # R Environment Variables .Renviron -# pkgdown site +# Locally generated pkgdown site +docs docs/ -# translation temp files -po/*~ - -# RStudio Connect folder -rsconnect/ +# Mac files +.DS_Store +doc +Meta diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md new file mode 100644 index 0000000..d985ea2 --- /dev/null +++ b/CODE_OF_CONDUCT.md @@ -0,0 +1,126 @@ +# Contributor Covenant Code of Conduct + +## Our Pledge + +We as members, contributors, and leaders pledge to make participation in our +community a harassment-free experience for everyone, regardless of age, body +size, visible or invisible disability, ethnicity, sex characteristics, gender +identity and expression, level of experience, education, socio-economic status, +nationality, personal appearance, race, caste, color, religion, or sexual +identity and orientation. + +We pledge to act and interact in ways that contribute to an open, welcoming, +diverse, inclusive, and healthy community. + +## Our Standards + +Examples of behavior that contributes to a positive environment for our +community include: + +* Demonstrating empathy and kindness toward other people +* Being respectful of differing opinions, viewpoints, and experiences +* Giving and gracefully accepting constructive feedback +* Accepting responsibility and apologizing to those affected by our mistakes, + and learning from the experience +* Focusing on what is best not just for us as individuals, but for the overall + community + +Examples of unacceptable behavior include: + +* The use of sexualized language or imagery, and sexual attention or advances of + any kind +* Trolling, insulting or derogatory comments, and personal or political attacks +* Public or private harassment +* Publishing others' private information, such as a physical or email address, + without their explicit permission +* Other conduct which could reasonably be considered inappropriate in a + professional setting + +## Enforcement Responsibilities + +Community leaders are responsible for clarifying and enforcing our standards of +acceptable behavior and will take appropriate and fair corrective action in +response to any behavior that they deem inappropriate, threatening, offensive, +or harmful. + +Community leaders have the right and responsibility to remove, edit, or reject +comments, commits, code, wiki edits, issues, and other contributions that are +not aligned to this Code of Conduct, and will communicate reasons for moderation +decisions when appropriate. + +## Scope + +This Code of Conduct applies within all community spaces, and also applies when +an individual is officially representing the community in public spaces. +Examples of representing our community include using an official e-mail address, +posting via an official social media account, or acting as an appointed +representative at an online or offline event. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be +reported to the community leaders responsible for enforcement at phgrosjean@sciviews.org. +All complaints will be reviewed and investigated promptly and fairly. + +All community leaders are obligated to respect the privacy and security of the +reporter of any incident. + +## Enforcement Guidelines + +Community leaders will follow these Community Impact Guidelines in determining +the consequences for any action they deem in violation of this Code of Conduct: + +### 1. Correction + +**Community Impact**: Use of inappropriate language or other behavior deemed +unprofessional or unwelcome in the community. + +**Consequence**: A private, written warning from community leaders, providing +clarity around the nature of the violation and an explanation of why the +behavior was inappropriate. A public apology may be requested. + +### 2. Warning + +**Community Impact**: A violation through a single incident or series of +actions. + +**Consequence**: A warning with consequences for continued behavior. No +interaction with the people involved, including unsolicited interaction with +those enforcing the Code of Conduct, for a specified period of time. This +includes avoiding interactions in community spaces as well as external channels +like social media. Violating these terms may lead to a temporary or permanent +ban. + +### 3. Temporary Ban + +**Community Impact**: A serious violation of community standards, including +sustained inappropriate behavior. + +**Consequence**: A temporary ban from any sort of interaction or public +communication with the community for a specified period of time. No public or +private interaction with the people involved, including unsolicited interaction +with those enforcing the Code of Conduct, is allowed during this period. +Violating these terms may lead to a permanent ban. + +### 4. Permanent Ban + +**Community Impact**: Demonstrating a pattern of violation of community +standards, including sustained inappropriate behavior, harassment of an +individual, or aggression toward or disparagement of classes of individuals. + +**Consequence**: A permanent ban from any sort of public interaction within the +community. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], +version 2.1, available at +. + +Community Impact Guidelines were inspired by +[Mozilla's code of conduct enforcement ladder][https://github.com/mozilla/inclusion]. + +For answers to common questions about this code of conduct, see the FAQ at +. Translations are available at . + +[homepage]: https://www.contributor-covenant.org diff --git a/DESCRIPTION b/DESCRIPTION new file mode 100644 index 0000000..f4e6f22 --- /dev/null +++ b/DESCRIPTION @@ -0,0 +1,41 @@ +Package: learnitprogress +Type: Package +Version: 0.9.0 +Title: Report Student Progress in 'LearnIt::R' Courses +Description: A Shiny app that connects to you LRS ("Learning Record Store" + database, as created and managed by the 'learnitdown' package) and displays a + report showing the completion of exercices and projects. This report is + intended for students to track their progress in the course. +Authors@R: c(person("Philippe", "Grosjean", role = c("aut", "cre"), + email = "phgrosjean@sciviews.org", + comment = c(ORCID = "0000-0002-2694-9471"))) +Maintainer: Philippe Grosjean +Depends: + R (>= 4.2.0) +Imports: + cowplot, + dplyr, + fs, + ggplot2, + mongolite, + RCurl, + rlang, + shiny, + utils +Suggests: + shinybusy, + shinythemes, + knitr, + rmarkdown, + spelling, + testthat (>= 3.0.0) +License: MIT + file LICENSE +URL: https://github.com/learnitr/learnitprogress, https://learnitr.github.io/learnitprogress/ +BugReports: https://github.com/learnitr/learnitprogress/issues +Roxygen: list(markdown = TRUE) +RoxygenNote: 7.2.3 +VignetteBuilder: knitr +Encoding: UTF-8 +Language: en-US +ByteCompile: yes +Config/testthat/edition: 3 diff --git a/LICENSE b/LICENSE index 03b3f77..f712e7c 100644 --- a/LICENSE +++ b/LICENSE @@ -1,21 +1,2 @@ -MIT License - -Copyright (c) 2024 learnitr - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. +YEAR: 2024 +COPYRIGHT HOLDER: Philippe Grosjean diff --git a/LICENSE.md b/LICENSE.md new file mode 100644 index 0000000..3302a3c --- /dev/null +++ b/LICENSE.md @@ -0,0 +1,21 @@ +# MIT License + +Copyright (c) 2024 Philippe Grosjean + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/NAMESPACE b/NAMESPACE new file mode 100644 index 0000000..84ac6c4 --- /dev/null +++ b/NAMESPACE @@ -0,0 +1,51 @@ +# Generated by roxygen2: do not edit by hand + +export(check_profile) +export(class_aa) +export(class_logins) +export(get_all_data) +export(get_data) +export(get_github_data) +export(get_grade_data) +export(get_grade_dir) +export(get_grid_data) +export(get_profile) +export(h5p_prog) +export(is_correct_query) +export(learnr_prog) +export(plot_grade) +export(plot_progression) +export(profile_from_query) +export(query_params_list) +export(report_progress) +export(run_progress_report) +export(shiny_prog) +export(user_login) +import(shiny) +importFrom(RCurl,url.exists) +importFrom(cowplot,theme_cowplot) +importFrom(dplyr,bind_rows) +importFrom(dplyr,filter) +importFrom(dplyr,full_join) +importFrom(dplyr,mutate) +importFrom(dplyr,select) +importFrom(fs,dir_exists) +importFrom(fs,dir_ls) +importFrom(fs,path) +importFrom(ggplot2,aes) +importFrom(ggplot2,coord_flip) +importFrom(ggplot2,geom_col) +importFrom(ggplot2,geom_hline) +importFrom(ggplot2,geom_line) +importFrom(ggplot2,geom_point) +importFrom(ggplot2,geom_smooth) +importFrom(ggplot2,geom_vline) +importFrom(ggplot2,ggplot) +importFrom(ggplot2,labs) +importFrom(ggplot2,position_nudge) +importFrom(ggplot2,scale_fill_brewer) +importFrom(ggplot2,scale_y_discrete) +importFrom(mongolite,mongo) +importFrom(rlang,.data) +importFrom(utils,URLencode) +importFrom(utils,browseURL) diff --git a/NEWS.md b/NEWS.md new file mode 100644 index 0000000..f8a6215 --- /dev/null +++ b/NEWS.md @@ -0,0 +1,3 @@ +# learnitprogress 0.9.0 + +- Code developed so far is now placed in the {learnitprogress} package. diff --git a/R/class.R b/R/class.R new file mode 100644 index 0000000..f29b32a --- /dev/null +++ b/R/class.R @@ -0,0 +1,153 @@ +# sdd1m, sdd1c, sdd2m, sdd2c, sdd3m, sdd4m, sdd5m + +#' Get a list of all user logins in a given class +#' +#' @param class The short identifier of the class +#' @param url The URL of the MongoDB database (LRS) +#' @param as.json If `TRUE`, return the result as a JSON string +#' +#' @return A character vector with the results +#' @export +class_logins <- function(class, url = getOption("learnitr.lrs_url"), + as.json = FALSE) { + # TODO: allow using this in a different context as SDD + query <- switch(class, + sdd1m = '{ "icflag": { "$in": ["S-BIOG-006", "S-BIOG-006,S-BIOG-015", "S-BIOG-027", "S-BIOG-027,S-BIOG-061"] }, "enrolled": true }', + sdd1c = '{ "icourse": "S-BIOG-921", "enrolled": true }', + sdd2m = '{ "icflag": { "$in": ["S-BIOG-015", "S-BIOG-006,S-BIOG-015", "S-BIOG-061", "S-BIOG-027,S-BIOG-061"] }, "enrolled": true }', + sdd2c = '{ "icourse": "S-BIOG-937-958-959", "enrolled": true }', + sdd3m = '{ "icourse": "S-BIOG-025", "enrolled": true }', + sdd4mold = '{ "icourse": "S-BIOG-043", "enrolled": true }', + sdd4m = '{ "icourse": "S-BIOG-077", "enrolled": true }', + stop("Only 'sdd1m/c', 'sdd2m/c', 'sdd3m', or 'sdd4m(old)' are recognized classes")) + mdb <- try(mongolite::mongo("users", url = url), silent = TRUE) + if (inherits(mdb, "try-error")) + stop("Error: impossible to connect to the users database") + res <- mdb$find(query, '{ "login" : true, "iemail" : true, "_id" : false }') + + email <- res$iemail + res <- res$login + + # Do we output the json string to be used in MongoDB directly? + if (isTRUE(as.json)) { + res2 <- paste(res, collapse = '", "') + res <- structure(paste0('{ "$in": ["', res2, '"] }'), + logins = res, n = length(res)) + } else { + res <- structure(res, email = email) + } + res +} + +# Get a selection of the apps that match a given aa +# (optionally restrict to a single module) +# tt <- paste0("A", formatC(1:12, width = 2, flag = "0")); tt +# grepl("^A0[1-6]", tt) +# grepl("^A(0[7-9])|(1[0-2])", tt) +#' @export +#' @rdname class_logins +#' @param aa The short identifier of the activity +#' @param module The module to restrict too, or `NULL` for all modules +class_aa <- function(aa, url = getOption("learnitr.lrs_url"), as.json = FALSE, + module = NULL) { + # This should be extracted from the database instead! + # 2021-2022: we also exclude XnnA that correspond to excluded H5P exercices + if (!is.null(module)) { + # We only look for exercises in a specific module + rx <- switch(aa, + sdd1mq1 = , + sdd1cq1 = , + sdd1mq2 = , + sdd1cq2 = , + sdd1mq3 = , + sdd1cq3 = '^A', + sdd2mq1 = , + sdd2cq1 = , + sdd2mq2 = , + sdd2cq2 = , + #sdd2cq3 = , + sdd2mq3 = , + sdd2cq3 = '^B', + sdd3mq1 = , + sdd3mq3 = '^C', + sdd4mq1 = '^D', + sdd5mq1 = '^E', + stop("Not implemented for this aa")) + rx <- paste0(rx, module, '[^A]') + if (isTRUE(as.json)) + rx <- paste0('{ "$regex": "', rx, '", "$options": "" }') + return(rx) + } + + # prerequisite sections and should not be counted here + if (isTRUE(as.json)) { + switch(aa, + sdd1mq1 = , + sdd1cq1 = '{ "$regex": "^A0[1-6][^A]", "$options": "" }', + sdd1mq2 = , + sdd1cq2 = '{ "$regex": "^A(0[7-9])|(1[0-2])[^A]", "$options": "" }', + sdd1mq3 = , + sdd1cq3 = '{ "$regex": "^A(0[1-9])|(1[0-2])[^A]", "$options": "" }', + sdd2mq1 = , + sdd2cq1 = '{ "$regex": "^B0[1-5][^A]", "$options": "" }', + sdd2mq2 = , + sdd2cq2 = '{ "$regex": "^B(0[6-9])|(10)[^A]", "$options": "" }', + #sdd2cq3 = '{ "$regex": "^B(09)|(1[0-2])[^A]", "$options": "" }', + sdd2mq3 = , + sdd2cq3 = '{ "$regex": "^B(0[1-9])|(10)[^A]", "$options": "" }', + sdd3mq1 = '{ "$regex": "^C0[1-6][^A]", "$options": "" }', + sdd3mq3 = '{ "$regex": "^C0[1-6][^A]", "$options": "" }', + sdd4mq1 = '{ "$regex": "^D0[1-6][^A]", "$options": "" }', + #sdd5mq1 = '{ "$regex": "^E0[1-4][^A]", "$options": "" }', + stop("Not implemented for this aa")) + } else { + switch(aa, + sdd1mq1 = , + sdd1cq1 = "^A0[1-6][^A]", + sdd1mq2 = , + sdd1cq2 = "^A(0[7-9])|(1[0-2])[^A]", + sdd1mq3 = , + sdd1cq3 = "^A(0[1-9])|(1[0-2])[^A]", + sdd2mq1 = , + sdd2cq1 = "^B0[1-5][^A]", + sdd2mq2 = , + sdd2cq2 = "^B(0[6-9])|(10)[^A]", + sdd2mq3 = , + sdd2cq3 = "^B(0[1-9])|(10)[^A]", + sdd3mq1 = "^C0[1-6][^A]", + sdd3mq3 = "^C0[1-6][^A]", + sdd4mq1 = "^D0[1-6][^A]", + #sdd5mq1 = "^E0[1-4][^A]", + stop("Not implemented for this aa")) + } +} + +# Get the user login, given its email +#' @export +#' @rdname class_logins +#' @param email The email of the student +user_login <- function(email, url = getOption("learnitr.lrs_url")) { + # Make xsure email is in lowercase + email <- tolower(email) + # Get the login associated with a given email address + # In case we have several login, return the most recent one + mdb <- try(mongolite::mongo("users", url = url), silent = TRUE) + if (inherits(mdb, "try-error")) + stop("Error: impossible to connect to the users database") + #query <- paste0('{ "iemail": "', email_moodle_case(email), '" }') + query <- paste0('{ "iemail": "', tolower(email), '" }') + fields <- '{ "login": true, "registered": true, "_id": false }' + if (!mdb$count(query)) + return(NULL) # This email is not found + res <- mdb$find(query, fields) + try(mdb$disconnect, silent = TRUE) # Not all versions have this! + # If there are several logins, return most recent one, and the list of all the + # others as an attribute + if (nrow(res) > 1) { + res <- res[order(res$registered, decreasing = TRUE), ] + logins <- res$login + structure(res$login[1], logins = logins) + } else {# Only one item + res$login + } +} diff --git a/R/github.R b/R/github.R new file mode 100644 index 0000000..5a95f4c --- /dev/null +++ b/R/github.R @@ -0,0 +1,15 @@ +# Check if a github user exists (just check if the URL exists, but could also +# be an organization and could not be indeed the user's account => check later) +#' Check a GitHub account +#' +#' @param user The user login to check on GitHub. +#' +#' @return `TRUE` if the user login exists on GitHub, `FALSE` otherwise. +#' @export +#' +#' @examples +#' is_github_user("phgrosjean") +#' is_github_user("nonexistinguser") +is_github_user <- function(user) { + RCurl::url.exists(paste0("https://github.com/", user, "/")) +} diff --git a/R/learnitprogress-package.R b/R/learnitprogress-package.R new file mode 100644 index 0000000..16fe7fa --- /dev/null +++ b/R/learnitprogress-package.R @@ -0,0 +1,24 @@ +#' @details +#' The learnitprogress package implements a Shiny application that summarizes +#' the progression of each of your course's students in their LearnIt::R +#' exercises and projects. It queries the MongoDB database that is collecting +#' your LearnIt::R activities (H5P, Shiny, learnr, GitHub projects...). +#' +#' See ... +#' @keywords internal +"_PACKAGE" + +#' @import shiny +#' @importFrom cowplot theme_cowplot +#' @importFrom dplyr bind_rows filter full_join mutate select +#' @importFrom fs dir_exists dir_ls path +#' @importFrom ggplot2 aes coord_flip geom_col geom_hline geom_line geom_point geom_smooth geom_vline ggplot labs position_nudge scale_fill_brewer scale_y_discrete +#' @importFrom mongolite mongo +#' @importFrom RCurl url.exists +#' @importFrom rlang .data +#' @importFrom utils browseURL URLencode +# The following block is used by usethis to automatically manage +# roxygen namespace tags. Modify with care! +## usethis namespace: start +## usethis namespace: end +NULL diff --git a/R/lrs.R b/R/lrs.R new file mode 100644 index 0000000..657feeb --- /dev/null +++ b/R/lrs.R @@ -0,0 +1,301 @@ +#' Get data from exercises or projects for one student +#' +#' @param collection The collection in the MongoDB database +#' @param query The JSON query on the MongoDB database +#' @param fields The fields to return +#' @param url The url to access the MongoDB database +#' @param count Do we count items instead of getting them? +#' +#' @return A data frame or a list with the data +#' @export +get_data <- function(collection, query, fields = '{}', + url = getOption("learnitr.lrs_url"), count = FALSE) { + mdb <- try(mongolite::mongo(collection, url = url), silent = TRUE) + if (inherits(mdb, "try-error")) { + stop("Error: impossible to connect to the database") + } else { + if (isTRUE(count)) { + res <- mdb$count(query) + } else { + res <- mdb$find(query, fields = fields) + } + } + mdb$disconnect() + res +} + +#' @export +#' @rdname get_data +#' @param email The email of the student +#' @param login The GitHub login of the student as alternate way to get its +#' profile (when it is not `NULL`) +get_profile <- function(email, login = NULL) { + if (is.null(login)) { + get_data("users", query = + paste0('{"iemail" : { "$eq" : "', tolower(email), '" } }'))[1, ] + } else { + get_data("users", query = + paste0('{"iemail" : { "$eq" : "', tolower(email), + '" }, "user_login": { "$eq": "', login, '" } }'))[1, ] + } +} + +#' @export +#' @rdname get_data +#' @param icourse The course identifier +#' @param module The module, or `NULL` for exercises in all modules +get_all_data <- function(email, icourse, module = NULL) { + message("Email: '", email, "'") + message("ICourse: '", icourse, "'") + if (!is.null(module)) + message("Module: '", module, "'") + if (is.null(email) || email == "") + return() + + # TODO: allow for using this outside of SDD courses + course <- "sdd1m" # Default for S-BIOG-006 & S-BIOG-027 + if (icourse == "S-BIOG-015" || icourse == "S-BIOG-061") course <- "sdd2m" + if (icourse == "S-BIOG-025") course <- "sdd3m" + if (icourse == "S-BIOG-921") course <- "sdd1c" + if (icourse == "S-BIOG-937-958-959") course <- "sdd2c" + message("Course: '", course, "'") + aa <- paste0(course, "q1") + # Special case for q2 courses + if (icourse == "S-BIOG-027") aa <- "sdd1mq2" + if (icourse == "S-BIOG-061") aa <- "sdd2mq2" + class_logs <- class_logins(course, as.json = TRUE) + message("Class query: ", class_logs) + class_apps <- class_aa(aa, as.json = TRUE, module = module) + user <- user_login(email) + message("User login: '", user, "'") + resh <- h5p_prog(user, class_logs, class_apps) + #print(nrow(resh)) + #print(head(resh)) + resl <- learnr_prog(user, class_logs, class_apps) + #print(nrow(resl)) + #print(head(resl)) + ress <- shiny_prog(user, class_logs, class_apps) + #print(nrow(ress)) + #print(head(ress)) + # We deal with all the possible cases of missing data + if (is.null(resh) && is.null(resl) && is.null(ress) ) + return(structure(list(), comment = "Pas encore d'enregistrements.")) + if (is.null(resh)) { + if (is.null(resl)) { + res <- ress + } else {# resl not null + if (is.null(ress)) { + res <- resl + } else { + res <- rbind(resl, ress) + } + } + } else {# resh not null + res <- resh + if (!is.null(resl)) + res <- rbind(res, resl) + if (!is.null(ress)) + res <- rbind(res, ress) + } + + res <- res[order(res$app), ] #res %<-% arrange(res, app) + # If not any student finish an exercise, we got NA in different places + # -> replace these by zero for the plot + res$max[is.na(res$max)] <- 0 + res$progress_max[is.na(res$progress_max)] <- 0 + res$progress[is.na(res$progress)] <- 0 + res$raw_score_max[is.na(res$raw_score_max)] <- 0 + res$raw_score_avg[is.na(res$raw_score_avg)] <- 0 + res$raw_score[is.na(res$raw_score)] <- 0 + res$count[is.na(res$count)] <- 0 + res$activity[is.na(res$activity)] <- 0 + attr(res, "user") <- user + res +} + +#' @export +#' @rdname get_data +get_github_data <- function(email) { + # Return NULL for now + NULL +} + +#' @export +#' @rdname get_data +get_grid_data <- function(email, icourse, module = NULL) { + message("Email: '", email, "'") + message("ICourse: '", icourse, "'") + if (!is.null(module)) + message("Module: '", module, "'") + if (is.null(email) || email == "") + return() + + course <- "sdd1m" # Default for S-BIOG-006 & S-BIOG-027 + if (icourse == "S-BIOG-015" || icourse == "S-BIOG-061") course <- "sdd2m" + if (icourse == "S-BIOG-025") course <- "sdd3m" + if (icourse == "S-BIOG-921") course <- "sdd1c" + if (icourse == "S-BIOG-937-958-959") course <- "sdd2c" + aa <- paste0(course, "q1") + # Special case for q2 courses + if (icourse == "S-BIOG-027") aa <- "sdd1mq2" + if (icourse == "S-BIOG-061") aa <- "sdd2mq2" + message("Course aa: '", aa, "'") + user <- user_login(email) + grade_dir <- get_grade_dir() + if (is.null(grade_dir)) + return() + # Correction grids are supposed to be in ///*.csv + grid_dir <- fs::path(grade_dir, aa, user) + message("Grids dir: '", grid_dir, "'") + if (!dir_exists(grid_dir)) + return() + grids <- dir_ls(grid_dir, glob = "*.csv") + message("Number of grids: '", length(grids), "'") + if (!length(grids)) + return() + names(grids) <- sub("\\.csv$", "", basename(grids)) + # Do we return only grids for one module? + if (!is.null(module)) { + grids_modules <- substring(names(grids), 2L, 3L) + grids <- grids[grids_modules == module] + if (!length(grids)) + return() + } + grids +} + +#' @export +#' @rdname get_data +get_grade_data <- function(email, icourse) { + message("Email: '", email, "'") + message("ICourse: '", icourse, "'") + if (is.null(email) || email == "") + return() + + course <- "sdd1m" # Default for S-BIOG-006 & S-BIOG-027 + if (icourse == "S-BIOG-015" || icourse == "S-BIOG-061") course <- "sdd2m" + if (icourse == "S-BIOG-025") course <- "sdd3m" + if (icourse == "S-BIOG-921") course <- "sdd1c" + if (icourse == "S-BIOG-937-958-959") course <- "sdd2c" + aa <- paste0(course, "q1") + # Special case for q2 courses + if (icourse == "S-BIOG-027") aa <- "sdd1mq2" + if (icourse == "S-BIOG-061") aa <- "sdd2mq2" + message("Course aa: '", aa, "'") + user <- user_login(email) + grade_dir <- get_grade_dir() + if (is.null(grade_dir)) + return() + # Data are supposed to be in //grade_.rds + grade_file <- fs::path(grade_dir, aa, paste0("grade_", user, ".rds")) + message("Grade file: '", grade_file, "'") + if (!file.exists(grade_file)) + return() + readRDS(grade_file) +} + +# TODO: allow to customise root_dir +#' @export +#' @rdname get_data +#' @param root_dir The root directory where the grades are stored. +#' @param acad_year The academic year to use, or `NULL` to use the latest one. +get_grade_dir <- function(root_dir = "/data1/grades", acad_year = NULL) { + if (!dir_exists(root_dir)) + return(NULL) + if (is.null(acad_year)) { + # Look for the latest subdir in alphabetic order and use it + sub_dirs <- dir_ls(root_dir,type = "directory") + if (!length(sub_dirs)) + return(NULL) + acad_year <- basename(sub_dirs[length(sub_dirs)]) + } + # Check that the full directory exists + grade_dir <- fs::path(root_dir, acad_year) + if (!dir.exists(grade_dir)) + return(NULL) + grade_dir +} + + +# Unused code for now ----------------------------------------------------- + +#correct <- function(x, name, old, new) { +# x[x[[name]] == old, name] <- new +# x +#} + +#correct_by_id_version <- function(x, id, version, name, new) { +# x[x$id == id & x$version == version, name] <- new +# x +#} + +#get_learnr_data <- function(email) { +# res <- get_data("learnr", +# query = +# paste0('{"email" : { "$eq" : "', email, '" }, "max" : { "$gt" : 0 } }'), +# fields = '{ "data" : false }' +# ) +# # # I need to correct XNNa_... -> XNNLa_... +# # res$app <- sub("^([A-E][0-9]{2})([a-z]_.+$)", "\\1L\\2", res$app) +# # # I need to correct B01La_rappel because it is indicated 22 questions, +# # # but there are 23 of them!!! +# # res[res$app == "B01La_rappel", "max"] <- 23 +# # Eliminate A00, B00, C00, ... +# res <- res[!grepl("^.00", res$app), ] +# # We keep only highest score for each ex (app_label) +# # So, sort score descending, make app_label, and keep only unique items +# res <- res[order(res$score, decreasing = TRUE), ] +# app_label <- paste(res$app, res$label, sep = "_") +# res <- res[!duplicated(app_label), ] +# res +#} + +#get_h5p_data <- function(email) { +# res <- get_data("h5p", +# query = +# paste0('{"email" : { "$eq" : "', tolower(email), +# '" }, "max" : { "$gt" : 0 } }'), +# fields = '{ "data" : false }' +# ) +# # Well, here names are a complete mess! Thus, lot of work to correct these! +# # Replace NA by "" in version +# res[is.na(res$version), "version"] <- "" +# #res <- correct_by_id_version(res, "8", "437dae39-40f6-4251-a056-f747a9888778", +# # "app", "A02Hb_R markdown/diff script markdown") +# #res <- correct_by_id_version(res, "8", "9d23d3a0-490c-450b-9cdd-e8d0927f28ed", +# # "app", "A02Hb_R markdown/chunk definition") +# #res <- correct_by_id_version(res, "8", "21444f3a-efed-4c92-99d6-f870628611f4", +# # "app", "A02Hb_R markdown/why chunk") +# #res <- correct_by_id_version(res, "8", "3f31e62e-9c9c-457f-9a7d-ad08060f3588", +# # "app", "") # This just collects results from chunk definition & why chunk +# #res <- correct_by_id_version(res, "10", "dc224f3d-32a7-43c8-a92a-c3d3c0694cfd", +# # "app", "A02Ha_nuage/questions") +# #res <- correct_by_id_version(res, "10", "", +# # "app", "A02Ha_nuage") +# # ... +# # Eliminate items where app == "" +# res <- res[res$app != "", ] +# # Sort by decreasing score, then eliminate duplicate app items +# res <- res[order(res$score, decreasing = TRUE), ] +# res <- res[!duplicated(res$app), ] +# # Completed items are redundant with answered items +# res <- res[res$verb != "completed", ] +# res +#} + +#get_shiny_data <- function(email) { +# res <- get_data("shiny", +# query = +# paste0('{"email" : { "$eq" : "', tolower(email), +# '" }, "max" : { "$gt" : 0 } }'), +# fields = '{ "data" : false }' +# ) +# # # Unfortunately, corrections are required for app names +# # res <- correct(res, "app", "A02a_limits", "A02Sa_limits") +# +# # Keep only highest score obtained for each item +# res <- res[order(res$version, decreasing = TRUE), ] +# res <- res[order(res$score, decreasing = TRUE), ] +# res <- res[!duplicated(res$app), ] +# res +#} diff --git a/R/plot_grade.R b/R/plot_grade.R new file mode 100644 index 0000000..b3ac963 --- /dev/null +++ b/R/plot_grade.R @@ -0,0 +1,55 @@ +#' Visualize details about the grade for one student +#' +#' @param data Grading data +#' @param show.name Indicate the name of the student in the title +#' @param give.note Indicate the final note in the title +#' +#' @return A ggplot object +#' @export +plot_grade <- function(data, show.name = TRUE, give.note = TRUE) { + email <- data[1, ]$email + name <- sub("\\.", " ", sub("@.+$", " ", email)) + login <- data[1, ]$login + score <- round(data[1, ]$score, 1) + max <- data[1, ]$max + + # First combine summary with details (leaving one empty line between the two) + scores <- bind_rows( + mutate(attr(data, "details"), app = paste0(" ", .data$app)), + data.frame(app = "---------------"), + mutate(attr(data, "summary"), score = .data$score / .data$max * 10, + max = 10, app = ifelse(.data$type == "challenge", "challenge", + paste(.data$type, "(all)"))) + ) + + if (isTRUE(show.name)) { + if (isTRUE(give.note)) { + subtitle <- paste0(name, " - ", score, "/", max) + } else { + subtitle <- paste0(name, " - ", "(pas encore de note)") + } + } else { + if (isTRUE(give.note)) { + subtitle <- paste0("Note globale : ", score, "/", max) + } else { + subtitle <- paste0("Note globale : ", "(pas encore de note)") + } + subtitle <- paste0("Note globale : ", score, "/", max) + } + + ggplot(data = scores, aes(x = .data$app, y = .data$max)) + + geom_hline(yintercept = c(0:max(data$max)), col = "gray") + + # White background + geom_col(fill = "white", col = "gray50", width = 0.9) + + # Add max + geom_col(aes(x = .data$app, y = .data$max), fill = "gray95", col = "gray50", + width = 0.9) + + # User scores + geom_col(aes(x = .data$app, y = .data$score, fill = .data$type), + col = "black", width = 0.9) + + coord_flip() + + scale_y_discrete(expand = c(0, 0)) + + cowplot::theme_cowplot(font_size = 10) + + labs(subtitle = subtitle, x = "", y = "") + + scale_fill_brewer(palette = "Dark2") +} diff --git a/R/plot_progression.R b/R/plot_progression.R new file mode 100644 index 0000000..5a28c4f --- /dev/null +++ b/R/plot_progression.R @@ -0,0 +1,33 @@ +#' Progression plot for one student (exercises H5P, Shiny & Learnr) +#' +#' @param user The user login +#' @param data The data for this user +#' +#' @return A ggplot object +#' +#' @export +plot_progression <- function(user, data) { + ggplot(data = data, aes(x = .data$app, y = .data$max)) + + geom_hline(yintercept = c(0:max(data$max)), col = "gray") + + # White background + geom_col(fill = "white", col = "gray50", width = 0.9) + + # Average and max progression (avg and max raw_score not shown here) + geom_col(aes(x = .data$app, y = .data$progress_max), fill = "gray95", + col = "gray50", width = 0.9) + + geom_col(aes(x = .data$app, y = .data$raw_score_avg), fill = "gray85", + col = "gray50", width = 0.9) + + # User's progression and raw_score + geom_col(aes(x = .data$app, y = .data$progress), fill = "red4", + col = "black", width = 0.5) + + geom_col(aes(x = .data$app, y = .data$raw_score), fill = "royalblue4", + col = "black", width = 0.5) + + # User's relative activity indicator + geom_col(aes(x = .data$app, y = .data$activity), fill = "black", + col = "black", width = 0.03, position = position_nudge(x = -0.45)) + + coord_flip() + + #scale_x_discrete(expand = c(0, 0)) + + scale_y_discrete(expand = c(0, 0)) + + cowplot::theme_cowplot(font_size = 14) + + labs(x = "", y = "") + #labs(title = paste("Progression -", user), x = "", y = "") +} diff --git a/R/progression.R b/R/progression.R new file mode 100644 index 0000000..92b8d72 --- /dev/null +++ b/R/progression.R @@ -0,0 +1,469 @@ +#' Progression in learnrs, H5P and Shiny apps +#' +#' @param user_login The GitHub login of one student +#' @param class_logins All the logins for the class +#' @param class_aa The AA identifier for this class +#' @param class_data The data for this class +#' @param url The url of the MongoDB data base +#' +#' @return A data frame with all the progression data for the user and the class +#' @export +learnr_prog <- function(user_login, class_logins, class_aa, class_data = NULL, + url = getOption("learnitr.lrs_url")) { + if (is.null(class_data)) { + class_data <- learnr_class_prog(class_logins, class_aa, url = url) + } + if (is.null(class_data)) + return(NULL) # No Shiny data for this course + user_data <- learnr_user_prog(user_login, class_aa, url = url) + if (is.null(user_data)) { + n <- nrow(class_data) + user_data <- data.frame(app = class_data$app, id = rep("", n), + items_done = rep("", n), count = rep(0, n), + progress = rep(0, n), raw_score = rep(0, n)) + } + + # Merge statistics for user and class by app + res <- dplyr::full_join(class_data, user_data) + res <- dplyr::select(res, 'app', 'id', 'items', 'items_done', 'max', + 'progress_max', 'progress', 'raw_score_max', 'raw_score_avg', 'raw_score', + 'count_all', 'count_avg', 'count') + res <- dplyr::mutate(res, + activity = pmin(1, .data$count / .data$count_avg) * + pmax(.data$max, na.rm = TRUE)) + as.data.frame(res) +} + +#' @export +#' @rdname learnr_prog +h5p_prog <- function(user_login, class_logins, class_aa, class_data = NULL, + url = getOption("learnitr.lrs_url")) { + if (is.null(class_data)) { + class_data <- h5p_class_prog(class_logins, class_aa, url = url) + } + if (is.null(class_data)) + return(NULL) # No Shiny data for this course + + user_data <- h5p_user_prog(user_login, class_aa, url = url) + if (is.null(user_data)) { + n <- nrow(class_data) + user_data <- data.frame(id = class_data$id, + items_done = rep("", n), count = rep(0, n), + progress = rep(0, n), raw_score = rep(0, n)) + } + + # Merge statistics for user and class by app + res <- dplyr::full_join(class_data, user_data) + res <- dplyr::select(res, 'app', 'id', 'items', 'items_done', 'max', + 'progress_max', 'progress', 'raw_score_max', 'raw_score_avg', 'raw_score', + 'count_all', 'count_avg', 'count') + res <- dplyr::mutate(res, + activity = pmin(1, .data$count / .data$count_avg) * + pmax(.data$max, na.rm = TRUE)) + as.data.frame(res) +} + +#' @export +#' @rdname learnr_prog +shiny_prog <- function(user_login, class_logins, class_aa, class_data = NULL, + url = getOption("learnitr.lrs_url")) { + if (is.null(class_data)) { + class_data <- shiny_class_prog(class_logins, class_aa, url = url) + } + if (is.null(class_data)) + return(NULL) # No Shiny data for this course + user_data <- shiny_user_prog(user_login, class_aa, url = url) + if (is.null(user_data)) { + n <- nrow(class_data) + user_data <- data.frame(app = class_data$app, id = rep("", n), + items = class_data$app, items_done = rep("", n), count = rep(0, n), + progress = rep(0, n), raw_score = rep(0, n)) + } + + # Merge statistics for user and class by app + res <- dplyr::full_join(class_data, user_data) + res <- dplyr::select(res, 'app', 'id', 'items', 'items_done', 'max', + 'progress_max', 'progress', 'raw_score_max', 'raw_score_avg', 'raw_score', + 'count_all', 'count_avg', 'count') + res <- dplyr::mutate(res, + activity = pmin(1, .data$count / .data$count_avg) * + pmax(.data$max, na.rm = TRUE)) + as.data.frame(res) +} + + +# Internal functions ------------------------------------------------------ + +# Get Learnr data for a whole class +learnr_class_prog <- function(class_logins, class_aa, + url = getOption("learnitr.lrs_url")) { + mdb_learnr <- try(mongolite::mongo("learnr", url = url), silent = TRUE) + if (inherits(mdb_learnr, "try-error")) + stop("Error: impossible to connect to the learnr database") + + if (!mdb_learnr$count(paste0('{ "login": ', class_logins, ', + "app": ', class_aa, ', "max": { "$gt": 0 }, + "verb": { "$in": ["answered", "submitted"] } }'))) + return(NULL) + + part1 <- mdb_learnr$aggregate(paste0('[ { + "$match": { + "login": ', class_logins, ', + "app": ', class_aa, ', + "max": { "$gt": 0 }, + "verb": { "$in": ["answered", "submitted"] } + } +}, { + "$group": { + "_id": "$app", + "count": { "$sum": 1 }, + "done": { "$addToSet": "$label" }, + "succeeded": { "$addToSet": { + "$cond": [ { "$eq": ["$score", 1] }, "$label", null ] + } }, + "max": { "$max": "$max" } + } +}, { + "$project": { + "app": "$_id", + "items": "$done", + "count_all": "$count", + "count_avg": { "$divide": ["$count", ', attr(class_logins, "n"), '] }, + "progress_max": { "$size": "$done" }, + "raw_score_max": { "$size": { "$setIntersection": ["$succeeded", "$done"] } }, + "max": "$max", + "_id": false + } +} ]')) + + # raw_score_avg + part2 <- mdb_learnr$aggregate(paste0('[ { + "$match": { + "login": ', class_logins, ', + "app": ', class_aa, ', + "max": { "$gt": 0 }, + "verb": { "$in": ["answered", "submitted"] } + } +}, { + "$group": { + "_id": { "login": "$login", "app": "$app" }, + "done": { "$addToSet": "$label" }, + "succeeded": { "$addToSet": { + "$cond": [ { "$eq": ["$score", 1]}, "$label", null ] + } } + } +}, { + "$project": { + "login": "$_id.login", + "app": "$_id.app", + "raw_score_max_by_user": { "$size": { "$setIntersection": ["$succeeded", "$done"] } }, + "_id": false + } +}, { + "$group": { + "_id": "$app", + "raw_score_avg": { "$avg": "$raw_score_max_by_user" } + } +}, { + "$project": { + "app": "$_id", + "raw_score_avg": { "$ifNull": [ "$raw_score_avg", 0] }, + "_id": false + } +} ]')) + dplyr::full_join(part1, part2) +} + +# Get learnrs progression for one student +learnr_user_prog <- function(user_login, class_aa, + url = getOption("learnitr.lrs_url")) { + mdb_learnr <- try(mongolite::mongo("learnr", url = url), silent = TRUE) + if (inherits(mdb_learnr, "try-error")) + stop("Error: impossible to connect to the learnr database") + + if (!mdb_learnr$count(paste0('{ "login": "', user_login, '", + "app": ', class_aa, ', "max": { "$gt": 0 }, + "verb": { "$in": ["answered", "submitted"] } }'))) + return(NULL) + + mdb_learnr$aggregate(paste0('[ { + "$match": { + "login": "', user_login, '", + "app": ', class_aa, ', + "max": { "$gt": 0 }, + "verb": { "$in": ["answered", "submitted"] } + } +}, { + "$group": { + "_id": "$app", + "count": { "$sum": 1 }, + "done" : { "$addToSet": "$label" }, + "succeeded": { "$addToSet": { + "$cond": [ { "$eq": ["$score", 1]}, "$label", null ] + } } + } +}, { + "$project": { + "app": "$_id", + "id": "", + "items_done": { "$setDifference": [ "$succeeded", [null] ] }, + "count": "$count", + "progress": { "$size": "$done" }, + "raw_score": { "$size": { "$setIntersection": ["$succeeded", "$done"] } }, + "_id": false + } +} ]')) +} + +# Get H5P data for a whole class +h5p_class_prog <- function(class_logins, class_aa, + url = getOption("learnitr.lrs_url")) { + mdb_h5p <- try(mongolite::mongo("h5p", url = url), silent = TRUE) + if (inherits(mdb_h5p, "try-error")) + stop("Error: impossible to connect to the H5P database") + + if (!mdb_h5p$count(paste0('{ "login": ', class_logins, ', + "app": ', class_aa, ' }'))) + return(NULL) + + part1 <- mdb_h5p$aggregate(paste0('[ { + "$match": { + "login": ', class_logins, ', + "app": ', class_aa, ' + } +}, { + "$group": { + "_id": "$id", + "apps": { "$addToSet": { + "$cond": [ { "$eq": ["$version", null] }, "$app", null ] + } }, + "count": { "$sum": { + "$cond": [ { "$in": ["$verb", ["attempted"]] }, 0, 1 ] + } } + } +}, { + "$project": { + "id": "$_id", + "app": { "$first": { "$setDifference": [ "$apps", [null] ] } }, + "count_all": "$count", + "count_avg": { "$divide": ["$count", ', attr(class_logins, "n"), '] }, + "_id": false + } +} ]')) + + part2 <- mdb_h5p$aggregate(paste0('[ { + "$match": { + "login": ', class_logins, ', + "app": ', class_aa, ', + "verb": "answered", + "app": { "$ne": "" }, + "max": { "$gt": 0 } + } +}, { + "$group": { + "_id": { "id": "$id", "app": "$app" }, + "score_max": { "$max": "$score" }, + "score_avg": { "$avg": "$score" }, + "max": { "$max": "$max" } + } +}, { + "$project": { + "id": "$_id.id", + "app": "$_id.app", + "progress": "$max", + "raw_score_max": "$score_max", + "raw_score_avg": "$score_avg", + "max": "$max", + "_id": false + } +}, { + "$group": { + "_id": "$id", + "items": { "$addToSet": "$app" }, + "progress": { "$sum": "$progress" }, + "raw_score_max": { "$sum": "$raw_score_max" }, + "raw_score_avg": { "$avg": "$raw_score_avg" }, + "max": { "$sum": "$max" } + } +}, { + "$project": { + "id": "$_id", + "items": "$items", + "progress_max": "$progress", + "raw_score_max": "$raw_score_max", + "raw_score_avg": "$raw_score_avg", + "max": "$max", + "_id": false + } +} ]')) + + res <- dplyr::full_join(part1, part2) + # Eliminate entries where count_all is zero + res <- dplyr::filter(res, .data$count_all > 0) + res +} + +# Get progression in H5P exercises for one student +h5p_user_prog <- function(user_login, class_aa, + url = getOption("learnitr.lrs_url")) { + mdb_h5p <- try(mongolite::mongo("h5p", url = url), silent = TRUE) + if (inherits(mdb_h5p, "try-error")) + stop("Error: impossible to connect to the H5P database") + + if (!mdb_h5p$count(paste0('{ "login": "', user_login, '", + "app": ', class_aa, ' }'))) + return(NULL) + + part1 <- mdb_h5p$aggregate(paste0('[ { + "$match": { + "login": "', user_login, '", + "app": ', class_aa, ' + } +}, { + "$group": { + "_id": "$id", + "count": { "$sum": { + "$cond": [ { "$in": ["$verb", ["attempted"]] }, 0, 1 ] + } } + } +}, { + "$project": { + "id": "$_id", + "count": "$count", + "_id": false + } +} ]')) + + if (!mdb_h5p$count(paste0('{ "login": "', user_login, '", + "app": ', class_aa, ', "verb": "answered", "max": { "$gt": 0 } }'))) { + # Fake data because the student did not answered to anything yet + n <- nrow(part1) + part2 <- data.frame(id = part1$id, items_done = rep("", n), + progress = rep(0, n), raw_score = rep(0, n)) + } else { + part2 <- mdb_h5p$aggregate(paste0('[ { + "$match": { + "login": "', user_login, '", + "app": ', class_aa, ', + "verb": "answered", + "max": { "$gt": 0 } + } +}, { + "$group": { + "_id": { "id": "$id", "app": "$app" }, + "progress": { "$max": "$max" }, + "raw_score": { "$max": "$score" } + } +}, { + "$project": { + "id": "$_id.id", + "app": "$_id.app", + "progress": "$progress", + "raw_score": "$raw_score", + "_id": false + } +}, { + "$group": { + "_id": "$id", + "items_done": { "$addToSet": "$app" }, + "progress": { "$sum": "$progress" }, + "raw_score": { "$sum": "$raw_score" } + } +}, { + "$project": { + "id": "$_id", + "items_done": "$items_done", + "progress": "$progress", + "raw_score": { "$ifNull": [ "$raw_score", 0] }, + "_id": false + } +} ]')) + } + + res <- dplyr::full_join(part1, part2) + # Eliminate entries where count is zero + res <- dplyr::filter(res, .data$count > 0) + res +} + +# Get Shiny apps data for a whole class +shiny_class_prog <- function(class_logins, class_aa, + url = getOption("learnitr.lrs_url")) { + mdb_shiny <- try(mongolite::mongo("shiny", url = url), silent = TRUE) + if (inherits(mdb_shiny, "try-error")) + stop("Error: impossible to connect to the shiny database") + + # Not used for now, but kept for future use... + #"progress_avg": { "$avg": { + # "$cond": [ { "$in": ["$verb", ["evaluated"] ] }, "$max", null ] + #} }, + + if (!mdb_shiny$count(paste0('{ "login": ', class_logins, ', + "app": ', class_aa, ' }'))) + return(NULL) + + mdb_shiny$aggregate(paste0('[ { + "$match": { + "login": ', class_logins, ', + "app": ', class_aa, ' + } +}, { + "$group": { + "_id": "$app", + "count": { "$sum": 1 }, + "raw_score_max": { "$max": "$score" }, + "raw_score_avg": { "$avg": "$score" }, + "max": { "$max": "$max" } + } +}, { + "$project": { + "app": "$_id", + "count_all": "$count", + "count_avg": { "$divide": ["$count", ', attr(class_logins, "n"), '] }, + "progress_max": "$max", + "raw_score_max": "$raw_score_max", + "raw_score_avg": "$raw_score_avg", + "max": "$max", + "_id": false + } +} ]')) +} + +# Get the progression in Shiny apps for one student +shiny_user_prog <- function(user_login, class_aa, + url = getOption("learnitr.lrs_url")) { + mdb_shiny <- try(mongolite::mongo("shiny", url = url), silent = TRUE) + if (inherits(mdb_shiny, "try-error")) + stop("Error: impossible to connect to the shiny database") + + if (!mdb_shiny$count(paste0('{ "login": "', user_login, '", + "app": ', class_aa, ' }'))) + return(NULL) + + mdb_shiny$aggregate(paste0('[ { + "$match": { + "login": "', user_login, '", + "app": ', class_aa, ' + } +}, { + "$group": { + "_id": "$app", + "count": { "$sum": 1 }, + "progress": { "$max": "$max" }, + "raw_score": { "$max": "$score" } + } +}, { + "$project": { + "app": "$_id", + "id": "", + "items": "$_id", + "items_done": { + "$cond": [ { "$gt": ["$progress", 0] }, ["$_id"], [] ] + }, + "count": "$count", + "progress": "$progress", + "raw_score": { "$ifNull": [ "$raw_score", 0] }, + "_id": false + } +} ]')) +} diff --git a/R/report_progress.R b/R/report_progress.R new file mode 100644 index 0000000..617cb1c --- /dev/null +++ b/R/report_progress.R @@ -0,0 +1,90 @@ +#' Show the progress report for one student +#' +#' @param email The email address of the student +#' @param second.course Should we display data for a secondary course that the +#' student follows? +#' @param course The primary course +#' @param module Should we restrict data for one module, (or all with `NULL`)? +#' @param browse Should we browse to the provided URL? +#' @param url The url to connect to the MongoDB database +#' @param port The port where the Shiny application is serving. If 0, a version +#' on a server (for instance, on Posit Connect) is used. +#' +#' @return The URL to the progress report +#' @export +#' +#' @examples +#' \dontrun{ +#' # In a different R session, run the following line: +#' learnitprogress::run_progress_report() +#' # Then... +#' options(learnitr.lrs_url = "mongodb://127.0.0.1/sdd") +#' learnitprogress::report_progress("student.email@school.edu") +#' } +report_progress <- function(email, second.course = FALSE, course = NULL, + module = NULL, browse = TRUE, url = getOption("learnitr.lrs_url"), + port = 3260) { + if (is.null(port)) + stop("The Shiny application progress report does not seem to be running") + # Get info about that user + # This is an alternate way to get user data that we use in SDD + #users <- data.io::read(svMisc::pcloud_crypto("sdd_users", "all_users.csv")) + #users <- drop_na(users, iemail) + # view_email <- tolower(email) + #user <- dplyr::filter(users, iemail == view_email) + mdb <- mongolite::mongo("users", url = url) + user <- mdb$find(paste0('{"iemail" : { "$eq" : "', tolower(email), '" } }')) + if (!NROW(user)) + stop("User not found!") + + # In user iemail, I have a lowercase version, but I need - + # email instead + user$iemail <- user$email + encode_param <- function(param, data = user) { + value <- URLencode(as.character(data[[param]]), reserved = TRUE) + paste0(param, "=", value) + } + params <- c("iemail", "iid", "ifirstname", "ilastname", "institution", + "icourse", "ictitle", "iurl", "iref") + # Possibly use first or second course from icflag + # Note: we don't update ictitle here => may be wrong! + if (!missing(second.course)) { + courses <- trimws(strsplit(user$icflag, ",", fixed = TRUE)[[1]]) + if (length(courses) > 1) { + if (isTRUE(second.course)) { + user[["icourse"]] <- courses[2] + } else {# First course + user[["icourse"]] <- courses[1] + } + } + } + search <- character(0) + for (param in params) + search <- c(search, encode_param(param)) + + # If course and module are specified, append them to the search string too + if (!is.null(course)) + search <- c(search, paste0("course=", course)) + if (!is.null(module)) + search <- c(search, paste0("module=", module)) + + if (port == 0) { + # We use the server version + # TODO: allow customizing this url + url <- paste0("https://sdd.umons.ac.be/sdd-progress-report/?", + paste(search, collapse = "&")) + } else { + # We use the local development version + url <- paste0("http://127.0.0.1:", port, "?", paste(search, collapse = "&")) + } + + if (isTRUE(browse)) + browseURL(url) + invisible(url) +} +#' @export +#' @rdname report_progress +run_progress_report <- function(port = 3260) { + shiny::runApp(system.file("shiny", "progress-report", + package = "learnitprogress"), port = port, launch.browser = FALSE) +} diff --git a/R/url.R b/R/url.R new file mode 100644 index 0000000..893a6c5 --- /dev/null +++ b/R/url.R @@ -0,0 +1,106 @@ +# Check and getURL parameters --------------------------------------------- + +#' Check student profile to identify it from URL's parameters +#' +#' @param profile Profile data gathered from the LRS +#' @param url_profile Profile data gathered from the URL +#' +#' @return `TRUE` if the profile is correct, `FALSE` otherwise. +#' @export +check_profile <- function(profile, url_profile) { + # In case of a wrong profile, return FALSE with a comment indicating what is + # wrong (stop at first error and don't check further) + incorrect <- function(comment) + structure(FALSE, comment = comment) + + # Github login not being in the URL params, it is not tested + # also, course and url are not tested as they may vary (a student may follow + # several courses, and the Moddle URL could change at any time) + if (tolower(profile$iemail) != tolower(as.character(url_profile['iemail']))) + return(incorrect("'iemail' incorrect")) + if (profile$iid != as.character(url_profile['iid'])) + return(incorrect("'iid' incorrect")) + if (tolower(profile$ifirstname) != + tolower(as.character(url_profile['ifirstname']))) + return(incorrect("'fist name' incorrect")) + if (tolower(profile$ilastname) != + tolower(as.character(url_profile['ilastname']))) + return(incorrect("'last name' incorrect")) + # Do not check institution because it is empty for a few students + #if (profile$institution != as.character(url_profile['institution'])) + # return(incorrect("'institution' incorrect")) + # Note: we don't check icourse and ictitle here, because a student may be + # registered to several courses => it may change from the stored profile + # For iref (Moodle user's UUID), this item is not accessible outside of Moodle + # (and our LRS once the user is registered). However, it makes problem for us, + # because we cannot check the report before a user registers. The magic value + # "teacher-special-access" for iref allows to by-pass that item (keep it?) + # It can be used both in the profile file and the URL for debugging purposes + #if (profile$iref != "teacher-special-access" && + # as.character(url_profile['iref']) != "teacher-special-access" && + # profile$iref != as.character(url_profile['iref'])) + # return(incorrect("'iref' incorrect")) + + # All test corrects => OK + TRUE +} + +#' URL parameters used to identify a student +#' +#' # Get the list of required parameters to identify a student from the URL. +#' +#' @return A character vector with the list of parameters +#' @details +#' In Moodle, the URL variables must be set as follows: +#' - iemail = Student's email +#' - iid = Student' ID ("Nom d'utilisateur" in French in Moodle) +#' - ifirstname = First name of the student +#' - ilastname = Last name of the student +#' - institution = Institution +#' - icourse = Identifier for the course +#' - ictitle = Title of the course +#' - iurl = URL of the Moodle server +#' - iref = Moodle ID +#' +#' @export +#' @rdname check_profile +#' +#' @examples +#' query_params_list() +query_params_list <- function() { + c("iemail", "iid", "ifirstname", "ilastname", + "institution", "icourse", "ictitle", "iurl", "iref") +} + +#' Check if all required parameters are in the query URL +#' +#' @param query The query part of the URL (text after ?, like in +#' https://mysite.org?params1=1¶m2=2). The query is a character string with +#' each value, named with the parameters names. +#' +#' @return `TRUE` if all required parameters are in the query, `FALSE` otherwise. +#' @export +#' @rdname check_profile +#' +#' @examples +#' is_correct_query(c(iemail = "student@uni.edu", iid = "1234", +#' ifirstname = "John", ilastname = "Doe", institution = "University", +#' icourse = "123", ictitle = "Course title", iurl = "https://moodle.uni.edu", +#' iref = "2FAB3EE8-42B5-4B61-DA23-2167FACD7B569")) +#' is_correct_query(c(iemail = "student@uni.edu")) +is_correct_query <- function(query) { + all(query_params_list() %in% names(query)) +} + +# +#' Reformat the query (eliminate possible other parameters) to get user profile +#' +#' @return A list with the required parameters +#' @export +#' @rdname check_profile +#' +#' @examples +#' profile_from_query(c(iemail = "student@uni.edu", iid = "1234", other = "a")) +profile_from_query <- function(query) { + query[query_params_list()] +} diff --git a/README.md b/README.md index 4b594b4..1eb676b 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,58 @@ -# learnitprogress -Shiny app showing the progression in the exercises to students +# learnitprogress - Report Student Progress in LearnIt::R Courses + + + +[![R-CMD-check](https://github.com/learnitr/learnitprogress/actions/workflows/R-CMD-check.yaml/badge.svg)](https://github.com/learnitr/learnitprogress/actions/workflows/R-CMD-check.yaml) +[![Codecov test coverage](https://img.shields.io/codecov/c/github/learnitr/learnitprogress/main.svg)](https://codecov.io/github/learnitr/learnitprogress?branch=main) +[![CRAN Status](https://www.r-pkg.org/badges/version/learnitprogress)](https://cran.r-project.org/package=learnitprogress) +[![r-universe status](https://learnitr.r-universe.dev/badges/learnitprogress)](https://learnitr.r-universe.dev/learnitprogress) +[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT) +[![Lifecycle stable](https://img.shields.io/badge/lifecycle-stable-brightgreen.svg)](https://lifecycle.r-lib.org/articles/stages.html#stable) + + + +A Shiny app that connects to you LRS ("Learning Record Store" database, as created and managed by the 'learnitdown' package) and displays a report showing the completion of exercices and projects. This report is intended for students to track their progress in the course. + +## Installation + +(Not yet! The latest stable version of {learnitprogress} can simply be installed from [CRAN](http://cran.r-project.org):) + +``` r +install.packages("learnitprogress") +``` + +You can also install the latest development version. Make sure you have the {remotes} R package installed: + +``` r +install.packages("remotes") +``` + +Use `install_github()` to install the {learnitprogress} package from GitHub (source from **main** branch will be recompiled on your machine): + +``` r +remotes::install_github("SciViews/learnitprogress") +``` + +R should install all required dependencies automatically, and then it should compile and install {learnitprogress}. + +## Further explore {learnitprogress} + +You can get further help about this package this way. Make the {learnitprogress} package available in your R session: + +``` r +library("learnitprogress") +``` + +Get help about this package: + +``` r +library(help = "learnitprogress") +help("learnitprogress-package") +vignette("learnitprogress") # None is installed with install_github() +``` + +For further instructions, please, refer to these help pages at . + +## Code of Conduct + +Please note that the {learnitprogress} project is released with a [Contributor Code of Conduct](CODE_OF_CONDUCT.md). By contributing to this project, you agree to abide by its terms. diff --git a/TODO.md b/TODO.md new file mode 100644 index 0000000..47297e7 --- /dev/null +++ b/TODO.md @@ -0,0 +1,7 @@ +# learnitprogress - To do list + +- A fake LRS as example. + +- Adapt the code to work also outside of SDD + +- English messages + translations (French...) diff --git a/_pkgdown.yml b/_pkgdown.yml new file mode 100644 index 0000000..4b28cb6 --- /dev/null +++ b/_pkgdown.yml @@ -0,0 +1,21 @@ +url: https://learnitr.github.io/learnitprogress + +destination: docs + +development: + mode: auto + +template: + params: + bootswatch: spacelab + mathjax: true + +toc: + depth: 3 + +home: + strip_header: true + +authors: + Philippe Grosjean: + href: https://phgrosjean.sciviews.org diff --git a/inst/CITATION b/inst/CITATION new file mode 100644 index 0000000..8035481 --- /dev/null +++ b/inst/CITATION @@ -0,0 +1,24 @@ +citHeader("To cite learnitr packages in publications use:") + +citEntry( + entry = "Manual", + title = "LearnIt::R", + author = personList(as.person("Philippe Grosjean"), + as.person("Guyliann Engels")), + organization = "UMONS", + address = "MONS, Belgium", + year = version$year, + url = "https://learnitr.r-universe.dev/", + + textVersion = + paste("Grosjean, Ph. & Engels, G. (", version$year, "). ", + "LearnIt::R. ", + "UMONS, Mons, Belgium. ", + "URL https://learnitr.r-universe.dev/.", + sep = "") +) + +citFooter("We have invested a lot of time and effort in creating LearnIt::R,", + "please cite it when using it together with R.", + "See also", sQuote("citation()"), + "for citing R.") diff --git a/inst/WORDLIST b/inst/WORDLIST new file mode 100644 index 0000000..fbd5364 --- /dev/null +++ b/inst/WORDLIST @@ -0,0 +1,28 @@ +CMD +Codecov +JSON +d'utilisateur +exercices +ggplot +https +icourse +ictitle +iemail +ifirstname +ilastname +iref +iurl +LearnIt +learnitdown +Learnr +learnr +learnrs +Lifecycle +LRS +Moodle +mysite +Nom +ORCID +param +params +URL's diff --git a/inst/shiny/progress-report/app.R b/inst/shiny/progress-report/app.R new file mode 100644 index 0000000..55b634a --- /dev/null +++ b/inst/shiny/progress-report/app.R @@ -0,0 +1,462 @@ +# Individual student's progress report for BioDataScience-Course at UMONS +# Version 3.0.0, Copyright (c), 2021-2024, Philippe Grosjean & Guyliann Engels +# +# TODO: +# - Rework to allow using outside of SDD context +# - Specify academic year +# - Check messages in the report +# - Better plots... may be using plotly for interactivity! + use median and max +# for learnr plots +# - Help pages /progress-login and /progress-report to implement + +# Need shinyFeedback v 0.3.0 from GitHub in svbox2020... OK for svbox2021 +#remotes::install_github("merlinoa/shinyFeedback@v0.3.0") + +#date_query_def <- '"date": { "$gt": "2022-09-18 00:00:00.000000" }, ' + +library(shiny) +#library(shinyjs) +library(shinybusy) +library(shinythemes) +#library(shinyFeedback) +library(RCurl) +library(mongolite) +#library(PKI) +library(dplyr) +library(ggplot2) +library(cowplot) +library(fs) +#library(svFlow) +library(learnitprogress) + + +# Connect to the course LRS (MongoDB database) ---------------------------- + +# TODO: make independent from SDD +options(learnitr.lrs_url = getOption("learnitr.lrs_url", + default = Sys.getenv("LEARNITR_LRS_URL", + unset = "mongodb://127.0.0.1/sdd"))) + +mdb_h5p <- mongolite::mongo('h5p', url = getOption("learnitr.lrs_url")) +mdb_h5p$disconnect() +mdb_learnr <- mongolite::mongo('learnr', url = getOption("learnitr.lrs_url")) +mdb_learnr$disconnect() +mdb_shiny <- mongolite::mongo('shiny', url = getOption("learnitr.lrs_url")) +mdb_shiny$disconnect() + + +# The Shiny app ----------------------------------------------------------- + +ui <- fluidPage(theme = shinytheme("lumen"), + titlePanel(textOutput("title"), windowTitle = "Progression"), + + # State #1: message when the app is launched with incorrect/incomplete URL + conditionalPanel('output.wrongURL !== null & output.wrongURL != ""', + span(tags$b(textOutput("wrongURL")), style = "color:red") + ), + +# # State #2: ask for login and user recording if not registered yet +# conditionalPanel('output.loginMessage !== null & output.loginMessage != ""', +# tags$b(textOutput("loginMessage")), +# shinyjs::useShinyjs(), +# shinyFeedback::useShinyFeedback(), +# textInput("login", "Login Github :", ""), +# actionButton("register", "Enregistrement", icon = icon("key")), +# helpText(htmlOutput("registerMessage")), +# tags$a("Page d'aide", +# href = "https://wp.sciviews.org/progress-login", target = "_blank") +# ), + + # State #3: display the progress report for the registered user + conditionalPanel('output.wrongURL == ""', # & output.loginMessage == ""', + + tabsetPanel( + tabPanel("Progression", + h4("Progression des H5Ps, Shiny apps & Learnrs"), + htmlOutput("reportMessage"), # General report message + htmlOutput("learnrMessage"), # Message specific for the learnrs + plotOutput("learnrPlot", height = 600), + + #h4("Progression sur Github"), + #htmlOutput("githubMessage"), # Message specific for Github activity + #plotOutput("githubPlot"), + tags$a("Page d'aide", + href = "https://wp.sciviews.org/progress-report") + ), + tabPanel("Projets GitHub", + h4("Grilles de correction des projets GitHub"), + htmlOutput("gridMessage"), # Message specific for the correction grids + selectInput("gridSelect", " ", c()), + dataTableOutput("gridTable"), + + tags$a("Page d'aide", + href = "https://wp.sciviews.org/progress-report") + ), + tabPanel("Note", + h4("D\u00e9tail de la note"), + htmlOutput("gradeMessage"), # Message specific for the grade plot + plotOutput("gradePlot", height = 600), + + tags$a("Page d'aide", + href = "https://wp.sciviews.org/progress-report") + ) + ) + ) +) + +server <- function(input, output, session) { + # User login, email, ... globally used in the app + user <- reactiveValues(data = NULL, learnr = NULL, github = NULL, + grade = NULL, grid = NULL, course = NULL, module = NULL, login = " ", + email = " ", firstname = " ", lastname = " ", message = "", + learnr_message = "", github_message = "", + grade_message = + "La note finale pour cette AA n'est pas encore attribu\u00e9e.", + grid_message = + "Pas de grille de correction de projet GitHub actuellement disponible.", + done = FALSE) + + # Modal busy box with spinner + shinybusy::show_modal_spinner(spin = "radar") + + # During calculations, login inputbox & register button are visible + # This was the only solution I found for now to avoid it +# shinyjs::hide("register") +# shinyjs::hide("login") + + # Get parameters transmitted by Moodle through the URL + observe({ + # Avoid displaying wrongURL and login conditional panels for now + query <- parseQueryString(session$clientData$url_search) + if (!is_correct_query(query)) {# Wrong URL + # Place (and lock down) the application in state #1 - nothing more to do + user$email <- "" # This way, conditionalPanel 'wrongURL' is activated + showNotification(type = "error", duration = 10, + "Le rapport de progression n'est disponible que depuis le cours (si on est correctement enregistr\u00e9) ou depuis Moodle.") + + } else if (!isTRUE(isolate(user$done))) {# Correct URL, process only once + user$course <- query$course + user$module <- query$module + user$data <- user_data <- profile_from_query(query) + user$firstname <- user_data['ifirstname'] + user$lastname <- user_data['ilastname'] + user_profile <- get_profile(user_data['iemail']) + # If there is an error, print it at the console + if (!is.null(comment(user_profile))) + cat("User profile error:", comment(user_profile), "\n") + + # if (!length(user_profile)) {# If the profile is empty + # # Switch in state #2: ask for user registration + # user$email <- user_data['iemail'] + # user$login <- "" + # shinyjs::show("login") + # shinyjs::show("register") + # + # } else {# Profile exists + # Check provided data match + res <- check_profile(user_profile, user_data) + if (!res) { + # Indicate what was wrong at the console + cat("Profile verification:", comment(res), "\n") + # Switch the app in state #1 (lock down) + user$email <- "" # This way, conditionalPanel 'wrongURL' is activated + showNotification(type = "error", duration = 10, + "Profil utilisateur incorrect, contactez vos enseignants.") + + } else {# Our app will be in state #3, create the progress report now + user$login <- user_profile$login + user$email <- user_profile$email + + # Get data for the learnrs and the Github activity + if (is.null(user$course)) { + course_message <- paste0("de ", user$data["ictitle"], " (", + user$data["icourse"], ")") + } else { + course_message <- paste(user$course) + } + if (is.null(user$module)) { + module_message <- "" + } else { + module_message <- paste(", module", user$module) + } + user$message <- paste0("Votre progression pour le cours ", + course_message, module_message, + ". Gris = progression de la classe, bleu = exercices r\u00e9ussis, rouge = r\u00e9ponses incorrectes. Ceci ne correspond pas \u00e0 la note finale mais seulement \u00e0 l'\u00e9tat d'avancement. Le rapport s'actualise toutes les 5 minutes (donc il se peut que vos derni\u00e8res actions ne soient pas encore visibles).") + + # Get learnr data + #user_learnr <- get_learnr_data(user$email) + if (is.null(user$course)) { + icourse <- user$data["icourse"] + } else { + icourse <- user$course + } + user_learnr <- get_all_data(user$data["iemail"], icourse, user$module) + if (!is.null(comment(user_learnr))) { + cat("Error getting Learnr, Shiny & H5P data:", comment(user_learnr), + "\n") + user$message <- paste0(user$message, + " Pas de donn\u00e9es Learnr, Shiny, H5P", + " ou donn\u00e9es erron\u00e9es.") + } else { + # TODO: get latest data from the database now! + user$learnr <- user_learnr + user$learnr_message <- user_learnr$comment + # Uncomment for debugging purposes + #user$learnr_message <- paste0("Nlines learnr data:", + # nrow(user_learnr$data)) + # Also get general message + if (!is.null(user_learnr$message)) + user$message <- paste0(user$message, "\n
\n", + user_learnr$message) + } + + # Get Github data + user_github <- get_github_data(user$email) + if (!is.null(comment(user_github))) { + cat("Error getting Github data:", comment(user_github), "\n") + user$message <- paste0(user$message, + " Pas de donn\u00e9es Github,", + " ou donn\u00e9es erron\u00e9es.") + } else { + user$github <- user_github + user$github_message <- user_github$comment + # Uncomment for debugging purposes + #user$github_message <- paste0("Nlines Github data:", + # nrow(user_github$data), + # " - Nlines trend: ", nrow(user_github$trend)) + } + + # Get grade data + user_grade <- get_grade_data(user$data["iemail"], icourse) + if (!is.null(user_grade)) { + user$grade <- user_grade + } + + # Get grid data + user_grid <- get_grid_data(user$data["iemail"], icourse, user$module) + if (!is.null(user_grid)) { + user$grid <- user_grid + updateSelectInput(session, "gridSelect", choices = user_grid, + selected = user_grid[1]) + } + } + #} + } + + # Avoid running these long calculations twice and close the busy modal box + user$done <- TRUE + shinybusy::remove_modal_spinner() + }) + + # App title, with user firstname and lastname if possible + output$title <- renderText({ + #paste("Progression", user$firstname, user$lastname) + # Don't put "Progression in the title, cf. already above in Moodle + paste(user$firstname, user$lastname) + }) + + # State #1: wrong URL parameters. Display only this message + output$wrongURL <- renderText({ + if (!is.null(user$email) && user$email == "") { + "Impossible d'afficher votre progression: vous devez lancer cette application depuis le cours et y ĂȘtre enregistrĂ© comme \u00e9tudiant ou depuis votre compte Moodle institutionnel. Si c'est le cas, voyez vos enseignants pour qu'ils corrigent le bug !" + } else "" + }) + +# # State #2: ask for Github login to register this user, no report yet +# output$loginMessage <- renderText({ +# if (!is.null(user$login) && user$login == "" && +# !is.null(user$email) && user$email != "") { +# "Votre rapport de progression n'est pas accessible car votre compte n'est pas encore valid\u00e9 ou est incorrect. Entrez votre login GitHub une fois votre compte GitHub cr\u00e9\u00e9, et cliquez sur 'Enregistrement'." +# } else "" +# }) +# +# observeEvent(input$login, +# shinyFeedback::feedbackWarning("login", input$login == "", +# "ComplĂ©tez votre login GitHub") +# ) +# +# response <- eventReactive(input$register, { +# cat("registering!\n") +# if (input$login == "") { +# +# # The message to display +# "Vous devez entrer votre login GitHub pour pouvoir vous enregistrer !" +# +# } else if (!is_github_user(input$login)) {# Non-existing Github account +# shinyFeedback::feedbackDanger("login", TRUE, +# "VĂ©rifiez votre login GitHub") +# +# # The message to display +# "Login GitHub incorrect (cr\u00e9er d'abord un compte GitHub)." +# +# } else {# Correct Github account +# shinyFeedback::feedbackSuccess("login", TRUE) +# +# # Create a record in the sdd/logins database table +# res <- record_sdd_login(input$login, user$data, +# role = "student", comment = "") +# if (res == "") {# OK, registering request recorded +# # Inactivate the register button + login +# shinyjs::disable("register") +# shinyjs::disable("login") +# #shinyjs::hide("register") +# #shinyjs::hide("login") +# +# # The message to display +# "Demande d'enregistrement effectu\u00e9e. Vous recevrez un mail quand il sera effectif." +# } else {# Error registering the user in the database +# +# # The message to display +# paste0("Login GitHub correct, mais probl\u00e8me lors de l'enregistrement : ", res, +# ". V\u00e9rifiez votre connexion Internet et r\u00e9essayez (\u00e9ventuellement plus tard).") +# } +# } +# }) +# +# output$registerMessage <- renderText({response()}) + + # State #3: display progress report + output$reportMessage <- renderText({ + if (!is.null(user$message)) { + user$message + } else "" + }) + + output$learnrMessage <- renderText({ + if (!is.null(user$learnr_message)) { + user$learnr_message + } else "" + }) + + output$learnrPlot <- renderPlot({ + if (!is.null(user$learnr)) { + #user_data <- user$learnr$data + user_data <- user$learnr + # user_data$`Mediane classe` <- user_data$median / user_data$max * 100 + # user_data$`Meilleur` <- user_data$best / user_data$max * 100 + # user_data$`Votre score` <- user_data$result / user_data$max * 100 + # user_data$tutorial <- factor(user_data$tutorial, + # levels = sort(unique(user_data$tutorial), decreasing = TRUE)) + # # TODO: ajouter les infos relatives Ă  la mĂ©diane de la classe et au meilleur + # res <- try(ggplot(user_data, aes(x = tutorial, y = `Votre score`, fill = score)) + + # #geom_col(aes(x = tutorial, y = `Mediane classe`, fill = "#888888")) + + # geom_col() + + # scale_fill_gradient(low = "firebrick", high = "lightblue") + + # coord_flip() + + # ylab("Pourcentage du tutoriel fait") + + # xlab("Tutoriel") + + # cowplot::theme_cowplot(font_size = 14)) + #res <- try(ggplot(user_data, aes(x = date, y = cum_max)) + + # geom_area() + + # xlab("Date") + + # ylab(("Nbre d'exercices")) + + # geom_area(aes(x = date, y = cum_grade), fill = "blue") + + # cowplot::theme_cowplot(font_size = 14) + res <- try(plot_progression(user$login, user_data)) + if (inherits(res, "try-error")) { + user$learnr_message <- as.character(res) + } else { + res + } + } + }) + + output$githubMessage <- renderText({ + if (!is.null(user$github_message)) { + user$github_message + } else "" + }) + + output$githubPlot <- renderPlot({ + if (!is.null(user$github)) { + user_data <- user$github$data + trend <- user$github$trend + res <- try(ggplot(user_data, aes(x = day, y = cusum)) + + geom_smooth(data = trend, aes(x = day, y = trend), + method = "loess", formula = y ~ x, + color = "#2e9ce6", alpha = 0.4, se = FALSE, span = 0.3) + + geom_line() + + geom_point() + + geom_vline(xintercept = as.Date("2020-03-15"), color = "red") + + geom_vline(xintercept = as.Date("2020-03-26"), color = "green") + + labs(y = "Somme des commits", x = "Temps", + #title = paste("ActivitĂ© de", student) , + caption = "Somme des commits au cours du temps en noir. La ligne bleue repr\u00e9sente l'activit\u00e9 moyenne de la classe. + Le trait vertical rouge repr\u00e9sente le d\u00e9but du confinement et le trait vertical vert marque la fin du dernier cours.") + + cowplot::theme_cowplot(font_size = 14)) + if (inherits(res, "try-error")) { + user$github_message <- as.character(res) + } else { + res + } + } + }) + + output$gradeMessage <- renderText({ + if (!is.null(user$grade_message)) { + user$grade_message + } else "" + }) + + output$gradePlot <- renderPlot({ + if (!is.null(user$grade)) { + user_grade <- user$grade + + res <- try(plot_grade(user_grade, show.name = FALSE)) + if (inherits(res, "try-error")) { + user$grade_message <- as.character(res) + } else { + if (is.null(user$course)) { + course_message <- paste0("de ", user$data["ictitle"], " (", + user$data["icourse"], ")") + } else { + course_message <- paste(user$course) + } + user$grade_message <- paste0("Notes obtenues par type d'exercice pour le cours ", + course_message, + ". Les notes peuvent \u00eatre inf\u00e9rieures \u00e0 la progression car elles tiennent compte des tentatives pour les H5P, des \u00e9l\u00e9ments corrects pour les applications Shiny et de la visualisation des aides pour les learnrs.") + res + } + } + }) + + output$gridMessage <- renderText({ + if (!is.null(user$grid_message)) { + user$grid_message + } else "" + }) + + output$gridTable <- renderDataTable({ + grid <- input$gridSelect + if (!is.null(grid) && !is.na(grid) && grid != "") { + tab <- try(read.csv(grid)) + if (inherits(tab, "try-error")) { + user$grid_message <- as.character(res) + } else { + # Get data for the learnrs and the Github activity + if (is.null(user$course)) { + course_message <- paste0("de ", user$data["ictitle"], " (", + user$data["icourse"], ")") + } else { + course_message <- paste(user$course) + } + if (is.null(user$module)) { + module_message <- "" + } else { + module_message <- paste(", module", user$module) + } + user$grid_message <- paste0("Evaluation de vos projets GitHub pour le cours ", + course_message, module_message, + ". Le crit\u00e8re indique l'\u00e9l\u00e9ment \u00e9valu\u00e9 (# titre =, @chunk =, !user =, YAML =, ...). Les notes sont cumulatives et certains crit\u00e8res peuvent servir \u00e0 diminuer la note globale ou pour un utilisateur en particulier (points n\u00e9gatifs).") + tab$note <- paste(round(tab$score, 2), tab$max, sep = "/") + tab$comment[is.na(tab$comment)] <- "" + tab <- tab[, c("file", "criterion", "note", "comment")] + names(tab) <- c("Fichier", "Crit\u00e8re", "Note", "Commentaire") + tab + } + } + }, options = list(pageLength = 100)) + +} + +shinyApp(ui = ui, server = server) diff --git a/learnitprogress.Rproj b/learnitprogress.Rproj new file mode 100644 index 0000000..fbbe594 --- /dev/null +++ b/learnitprogress.Rproj @@ -0,0 +1,24 @@ +Version: 1.0 + +RestoreWorkspace: Default +SaveWorkspace: Default +AlwaysSaveHistory: Default + +EnableCodeIndexing: Yes +UseSpacesForTab: Yes +NumSpacesForTab: 2 +Encoding: UTF-8 + +RnwWeave: knitr +LaTeX: XeLaTeX + +AutoAppendNewline: Yes +StripTrailingWhitespace: Yes + +BuildType: Package +PackageUseDevtools: Yes +PackageInstallArgs: --no-multiarch --with-keep.source +PackageCheckArgs: --as-cran +PackageRoxygenize: rd,collate,namespace,vignette + +SpellingDictionary: en_US diff --git a/man/check_profile.Rd b/man/check_profile.Rd new file mode 100644 index 0000000..8f5889c --- /dev/null +++ b/man/check_profile.Rd @@ -0,0 +1,61 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/url.R +\name{check_profile} +\alias{check_profile} +\alias{query_params_list} +\alias{is_correct_query} +\alias{profile_from_query} +\title{Check student profile to identify it from URL's parameters} +\usage{ +check_profile(profile, url_profile) + +query_params_list() + +is_correct_query(query) + +profile_from_query(query) +} +\arguments{ +\item{profile}{Profile data gathered from the LRS} + +\item{url_profile}{Profile data gathered from the URL} + +\item{query}{The query part of the URL (text after ?, like in +https://mysite.org?params1=1¶m2=2). The query is a character string with +each value, named with the parameters names.} +} +\value{ +\code{TRUE} if the profile is correct, \code{FALSE} otherwise. + +A character vector with the list of parameters + +\code{TRUE} if all required parameters are in the query, \code{FALSE} otherwise. + +A list with the required parameters +} +\description{ +# Get the list of required parameters to identify a student from the URL. +} +\details{ +In Moodle, the URL variables must be set as follows: +\itemize{ +\item iemail = Student's email +\item iid = Student' ID ("Nom d'utilisateur" in French in Moodle) +\item ifirstname = First name of the student +\item ilastname = Last name of the student +\item institution = Institution +\item icourse = Identifier for the course +\item ictitle = Title of the course +\item iurl = URL of the Moodle server +\item iref = Moodle ID +} +} +\examples{ +query_params_list() +is_correct_query(c(iemail = "student@uni.edu", iid = "1234", + ifirstname = "John", ilastname = "Doe", institution = "University", + icourse = "123", ictitle = "Course title", iurl = "https://moodle.uni.edu", + iref = "2FAB3EE8-42B5-4B61-DA23-2167FACD7B569")) +is_correct_query(c(iemail = "student@uni.edu")) +profile_from_query(c(iemail = "student@uni.edu", iid = "1234", other = "a")) +} diff --git a/man/class_logins.Rd b/man/class_logins.Rd new file mode 100644 index 0000000..008b6b3 --- /dev/null +++ b/man/class_logins.Rd @@ -0,0 +1,38 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/class.R +\name{class_logins} +\alias{class_logins} +\alias{class_aa} +\alias{user_login} +\title{Get a list of all user logins in a given class} +\usage{ +class_logins(class, url = getOption("learnitr.lrs_url"), as.json = FALSE) + +class_aa( + aa, + url = getOption("learnitr.lrs_url"), + as.json = FALSE, + module = NULL +) + +user_login(email, url = getOption("learnitr.lrs_url")) +} +\arguments{ +\item{class}{The short identifier of the class} + +\item{url}{The URL of the MongoDB database (LRS)} + +\item{as.json}{If \code{TRUE}, return the result as a JSON string} + +\item{aa}{The short identifier of the activity} + +\item{module}{The module to restrict too, or \code{NULL} for all modules} + +\item{email}{The email of the student} +} +\value{ +A character vector with the results +} +\description{ +Get a list of all user logins in a given class +} diff --git a/man/get_data.Rd b/man/get_data.Rd new file mode 100644 index 0000000..558567c --- /dev/null +++ b/man/get_data.Rd @@ -0,0 +1,62 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/lrs.R +\name{get_data} +\alias{get_data} +\alias{get_profile} +\alias{get_all_data} +\alias{get_github_data} +\alias{get_grid_data} +\alias{get_grade_data} +\alias{get_grade_dir} +\title{Get data from exercises or projects for one student} +\usage{ +get_data( + collection, + query, + fields = "{}", + url = getOption("learnitr.lrs_url"), + count = FALSE +) + +get_profile(email, login = NULL) + +get_all_data(email, icourse, module = NULL) + +get_github_data(email) + +get_grid_data(email, icourse, module = NULL) + +get_grade_data(email, icourse) + +get_grade_dir(root_dir = "/data1/grades", acad_year = NULL) +} +\arguments{ +\item{collection}{The collection in the MongoDB database} + +\item{query}{The JSON query on the MongoDB database} + +\item{fields}{The fields to return} + +\item{url}{The url to access the MongoDB database} + +\item{count}{Do we count items instead of getting them?} + +\item{email}{The email of the student} + +\item{login}{The GitHub login of the student as alternate way to get its +profile (when it is not \code{NULL})} + +\item{icourse}{The course identifier} + +\item{module}{The module, or \code{NULL} for exercises in all modules} + +\item{root_dir}{The root directory where the grades are stored.} + +\item{acad_year}{The academic year to use, or \code{NULL} to use the latest one.} +} +\value{ +A data frame or a list with the data +} +\description{ +Get data from exercises or projects for one student +} diff --git a/man/learnitprogress-package.Rd b/man/learnitprogress-package.Rd new file mode 100644 index 0000000..5ab781a --- /dev/null +++ b/man/learnitprogress-package.Rd @@ -0,0 +1,32 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/learnitprogress-package.R +\docType{package} +\name{learnitprogress-package} +\alias{learnitprogress} +\alias{learnitprogress-package} +\title{learnitprogress: Report Student Progress in 'LearnIt::R' Courses} +\description{ +A Shiny app that connects to you LRS ("Learning Record Store" database, as created and managed by the 'learnitdown' package) and displays a report showing the completion of exercices and projects. This report is intended for students to track their progress in the course. +} +\details{ +The learnitprogress package implements a Shiny application that summarizes +the progression of each of your course's students in their LearnIt::R +exercises and projects. It queries the MongoDB database that is collecting +your LearnIt::R activities (H5P, Shiny, learnr, GitHub projects...). + +See ... +} +\seealso{ +Useful links: +\itemize{ + \item \url{https://github.com/learnitr/learnitprogress} + \item \url{https://learnitr.github.io/learnitprogress/} + \item Report bugs at \url{https://github.com/learnitr/learnitprogress/issues} +} + +} +\author{ +\strong{Maintainer}: Philippe Grosjean \email{phgrosjean@sciviews.org} (\href{https://orcid.org/0000-0002-2694-9471}{ORCID}) + +} +\keyword{internal} diff --git a/man/learnr_prog.Rd b/man/learnr_prog.Rd new file mode 100644 index 0000000..f6e27db --- /dev/null +++ b/man/learnr_prog.Rd @@ -0,0 +1,49 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/progression.R +\name{learnr_prog} +\alias{learnr_prog} +\alias{h5p_prog} +\alias{shiny_prog} +\title{Progression in learnrs, H5P and Shiny apps} +\usage{ +learnr_prog( + user_login, + class_logins, + class_aa, + class_data = NULL, + url = getOption("learnitr.lrs_url") +) + +h5p_prog( + user_login, + class_logins, + class_aa, + class_data = NULL, + url = getOption("learnitr.lrs_url") +) + +shiny_prog( + user_login, + class_logins, + class_aa, + class_data = NULL, + url = getOption("learnitr.lrs_url") +) +} +\arguments{ +\item{user_login}{The GitHub login of one student} + +\item{class_logins}{All the logins for the class} + +\item{class_aa}{The AA identifier for this class} + +\item{class_data}{The data for this class} + +\item{url}{The url of the MongoDB data base} +} +\value{ +A data frame with all the progression data for the user and the class +} +\description{ +Progression in learnrs, H5P and Shiny apps +} diff --git a/man/plot_grade.Rd b/man/plot_grade.Rd new file mode 100644 index 0000000..29c3ef8 --- /dev/null +++ b/man/plot_grade.Rd @@ -0,0 +1,21 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/plot_grade.R +\name{plot_grade} +\alias{plot_grade} +\title{Visualize details about the grade for one student} +\usage{ +plot_grade(data, show.name = TRUE, give.note = TRUE) +} +\arguments{ +\item{data}{Grading data} + +\item{show.name}{Indicate the name of the student in the title} + +\item{give.note}{Indicate the final note in the title} +} +\value{ +A ggplot object +} +\description{ +Visualize details about the grade for one student +} diff --git a/man/plot_progression.Rd b/man/plot_progression.Rd new file mode 100644 index 0000000..3cb6018 --- /dev/null +++ b/man/plot_progression.Rd @@ -0,0 +1,19 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/plot_progression.R +\name{plot_progression} +\alias{plot_progression} +\title{Progression plot for one student (exercises H5P, Shiny & Learnr)} +\usage{ +plot_progression(user, data) +} +\arguments{ +\item{user}{The user login} + +\item{data}{The data for this user} +} +\value{ +A ggplot object +} +\description{ +Progression plot for one student (exercises H5P, Shiny & Learnr) +} diff --git a/man/report_progress.Rd b/man/report_progress.Rd new file mode 100644 index 0000000..01a1da5 --- /dev/null +++ b/man/report_progress.Rd @@ -0,0 +1,51 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/report_progress.R +\name{report_progress} +\alias{report_progress} +\alias{run_progress_report} +\title{Show the progress report for one student} +\usage{ +report_progress( + email, + second.course = FALSE, + course = NULL, + module = NULL, + browse = TRUE, + url = getOption("learnitr.lrs_url"), + port = 3260 +) + +run_progress_report(port = 3260) +} +\arguments{ +\item{email}{The email address of the student} + +\item{second.course}{Should we display data for a secondary course that the +student follows?} + +\item{course}{The primary course} + +\item{module}{Should we restrict data for one module, (or all with \code{NULL})?} + +\item{browse}{Should we browse to the provided URL?} + +\item{url}{The url to connect to the MongoDB database} + +\item{port}{The port where the Shiny application is serving. If 0, a version +on a server (for instance, on Posit Connect) is used.} +} +\value{ +The URL to the progress report +} +\description{ +Show the progress report for one student +} +\examples{ +\dontrun{ +# In a different R session, run the following line: +learnitprogress::run_progress_report() +# Then... +options(learnitr.lrs_url = "mongodb://127.0.0.1/sdd") +learnitprogress::report_progress("student.email@school.edu") +} +} diff --git a/pkgdown/favicon/apple-touch-icon-120x120.png b/pkgdown/favicon/apple-touch-icon-120x120.png new file mode 100755 index 0000000..a22e98d Binary files /dev/null and b/pkgdown/favicon/apple-touch-icon-120x120.png differ diff --git a/pkgdown/favicon/apple-touch-icon-152x152.png b/pkgdown/favicon/apple-touch-icon-152x152.png new file mode 100755 index 0000000..660d186 Binary files /dev/null and b/pkgdown/favicon/apple-touch-icon-152x152.png differ diff --git a/pkgdown/favicon/apple-touch-icon-180x180.png b/pkgdown/favicon/apple-touch-icon-180x180.png new file mode 100755 index 0000000..0a80eeb Binary files /dev/null and b/pkgdown/favicon/apple-touch-icon-180x180.png differ diff --git a/pkgdown/favicon/apple-touch-icon-60x60.png b/pkgdown/favicon/apple-touch-icon-60x60.png new file mode 100755 index 0000000..c4c9c21 Binary files /dev/null and b/pkgdown/favicon/apple-touch-icon-60x60.png differ diff --git a/pkgdown/favicon/apple-touch-icon-76x76.png b/pkgdown/favicon/apple-touch-icon-76x76.png new file mode 100755 index 0000000..7585083 Binary files /dev/null and b/pkgdown/favicon/apple-touch-icon-76x76.png differ diff --git a/pkgdown/favicon/apple-touch-icon.png b/pkgdown/favicon/apple-touch-icon.png new file mode 100755 index 0000000..0a80eeb Binary files /dev/null and b/pkgdown/favicon/apple-touch-icon.png differ diff --git a/pkgdown/favicon/favicon-16x16.png b/pkgdown/favicon/favicon-16x16.png new file mode 100755 index 0000000..015fcb3 Binary files /dev/null and b/pkgdown/favicon/favicon-16x16.png differ diff --git a/pkgdown/favicon/favicon-32x32.png b/pkgdown/favicon/favicon-32x32.png new file mode 100644 index 0000000..44582e9 Binary files /dev/null and b/pkgdown/favicon/favicon-32x32.png differ diff --git a/pkgdown/favicon/favicon.ico b/pkgdown/favicon/favicon.ico new file mode 100644 index 0000000..3a22db2 Binary files /dev/null and b/pkgdown/favicon/favicon.ico differ diff --git a/tests/spelling.R b/tests/spelling.R new file mode 100644 index 0000000..14bc897 --- /dev/null +++ b/tests/spelling.R @@ -0,0 +1,3 @@ +if (requireNamespace('spelling', quietly = TRUE)) + spelling::spell_check_test(vignettes = TRUE, error = FALSE, + skip_on_cran = TRUE) diff --git a/tests/testthat.R b/tests/testthat.R new file mode 100644 index 0000000..0b883c6 --- /dev/null +++ b/tests/testthat.R @@ -0,0 +1,13 @@ +# This file is part of the standard setup for testthat. +# It is recommended that you do not modify it. +# +# Where should you do additional test configuration? +# Learn more about the roles of various files in: +# * https://r-pkgs.org/testing-design.html#sec-tests-files-overview +# * https://testthat.r-lib.org/articles/special-files.html + +library(testthat) +library(learnitprogress) + + +test_check("learnitprogress") diff --git a/tests/testthat/test-lrs.R b/tests/testthat/test-lrs.R new file mode 100644 index 0000000..1a4bb48 --- /dev/null +++ b/tests/testthat/test-lrs.R @@ -0,0 +1,4 @@ +test_that("Connection to LRS is successful", { + # TODO: une an example database (or use data frames instead?) + expect_true(TRUE) +}) diff --git a/vignettes/.gitignore b/vignettes/.gitignore new file mode 100644 index 0000000..097b241 --- /dev/null +++ b/vignettes/.gitignore @@ -0,0 +1,2 @@ +*.html +*.R diff --git a/vignettes/learnitprogress.Rmd b/vignettes/learnitprogress.Rmd new file mode 100644 index 0000000..f9c6b81 --- /dev/null +++ b/vignettes/learnitprogress.Rmd @@ -0,0 +1,25 @@ +--- +title: "learnitprogress - Report Student Progress in 'LearnIt::R' Courses" +author: "Philippe Grosjean (phgrosjean@sciviews.org)" +date: "`r Sys.Date()`" +output: + rmarkdown::html_vignette: + fig_caption: yes +vignette: > + %\VignetteIndexEntry{learnitprogress - Report Student Progress in 'LearnIt::R' Courses} + %\VignetteEngine{knitr::rmarkdown} + %\VignetteEncoding{UTF-8} +--- + +```{r, include = FALSE} +knitr::opts_chunk$set( + collapse = TRUE, + comment = "#>" +) +``` + +```{r setup} +library(learnitprogress) +``` + +TODO...