From bec4378c53d5425fa5a1ea2003c7f17cdd51cc20 Mon Sep 17 00:00:00 2001 From: Taras Zakharko Date: Fri, 17 Dec 2021 10:11:46 +0100 Subject: [PATCH 1/2] Proof of concept implementation for local package caching Approach: relies on built-in caching for installed packages to store file checksums for local packages and use them to determine whether the contents have changed. --- R/type-local.R | 58 ++++++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 56 insertions(+), 2 deletions(-) diff --git a/R/type-local.R b/R/type-local.R index 6b3a51e2..705a0992 100644 --- a/R/type-local.R +++ b/R/type-local.R @@ -17,9 +17,34 @@ parse_remote_local <- function(specs, config, ...) { resolve_remote_local <- function(remote, direct, config, cache, dependencies, ...) { + # prepare a default resolution sources <- paste0("file://", normalizePath(remote$path, mustWork = FALSE)) - resolve_from_description(remote$path, sources, remote, direct, + res <- resolve_from_description(remote$path, sources, remote, direct, config, cache, dependencies[[2 - direct]]) + + # save the source as the remote url + res$metadata[["RemoteUrl"]] <- sources[[1]] + + # collect the list of local package files + pkg_files <- list.files( + remote$path[[1]], + full.names = TRUE, + recursive = TRUE, + # this will exclude hidden files, but maybe + # some package rely on them? + all.files = FALSE + ) + + # save the md5sum + # TODO: make one sum out of it? + # TODO: use sha256 and digest? + if (length(pkg_files) > 0) { + md5sum <- paste0(tools::md5sum(pkg_files), collapse = ".") + res$extra[[1]][["remotemd5sum"]] <- md5sum + res$metadata[["RemoteMD5Sum"]] <- md5sum + } + + res } download_remote_local <- function(resolution, target, target_tree, config, @@ -45,7 +70,36 @@ download_remote_local <- function(resolution, target, target_tree, config, } satisfy_remote_local <- function(resolution, candidate, config, ...) { - ## TODO: we can probably do better than this + ## 1. package name must match + if (resolution$package != candidate$package) { + return(structure(FALSE, reason = "Package names differ")) + } + + ## 2. installed from the same identical local source is good + if (candidate$type == "installed") { + want_reinst <- is_true_param(resolution$params[[1]], "reinstall") + if (want_reinst) { + return(structure(FALSE, reason = "Re-install requested")) + } + # check if the file path matches + candidate_url <- candidate$extra[[1]][["remoteurl"]] %||% NA_character_ + local_url <- resolution$sources[[1]] + if (!identical(candidate_url, local_url)) { + return(structure(FALSE, reason = "Installed package path mismatch")) + } + + # check if the md5sum maches + candidate_md5 <- candidate$extra[[1]][["remotemd5sum"]] %||% NA_character_ + local_md5 <- resolution$extra[[1]][["remotemd5sum"]] %||% NA_character_ + if (!identical(candidate_md5, local_md5)) { + return(structure(FALSE, reason = "Installed package md5sum mismatch")) + } + + # it's good! + return(TRUE) + } + + ## 3. no other candidate works FALSE } From 4c019b4bdd48597bbde2985c0ac91b747e158edf Mon Sep 17 00:00:00 2001 From: Taras Zakharko Date: Fri, 17 Dec 2021 13:48:33 +0100 Subject: [PATCH 2/2] Temporarily disable the test rendered incorrect with local caching --- tests/testthat/test-type-local.R | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/tests/testthat/test-type-local.R b/tests/testthat/test-type-local.R index ff2085a6..580dd9a4 100644 --- a/tests/testthat/test-type-local.R +++ b/tests/testthat/test-type-local.R @@ -155,7 +155,9 @@ test_that("download_remote error", { expect_true(all(dl$download_status == "Failed")) }) -test_that("satisfy", { - ## Always FALSE, independently of arguments - expect_false(satisfy_remote_local()) -}) +# remove this test for now +# +# test_that("satisfy", { +# ## Always FALSE, independently of arguments +# expect_false(satisfy_remote_local()) +# })